1 /*
2 FUSE: Filesystem in Userspace
3 Copyright (C) 2001-2007 Miklos Szeredi <miklos@szeredi.hu>
4
5 Implementation of the high-level FUSE API on top of the low-level
6 API.
7
8 This program can be distributed under the terms of the GNU LGPLv2.
9 See the file COPYING.LIB
10 */
11
12
13 /* For pthread_rwlock_t */
14 #define _GNU_SOURCE
15
16 #include "config.h"
17 #include "fuse_i.h"
18 #include "fuse_lowlevel.h"
19 #include "fuse_opt.h"
20 #include "fuse_misc.h"
21 #include "fuse_kernel.h"
22
23 #include <stdio.h>
24 #include <string.h>
25 #include <stdlib.h>
26 #include <stddef.h>
27 #include <stdbool.h>
28 #include <unistd.h>
29 #include <time.h>
30 #include <fcntl.h>
31 #include <limits.h>
32 #include <errno.h>
33 #include <signal.h>
34 #include <dlfcn.h>
35 #include <assert.h>
36 #include <poll.h>
37 #include <sys/param.h>
38 #include <sys/uio.h>
39 #include <sys/time.h>
40 #include <sys/mman.h>
41 #include <sys/file.h>
42
43 #define FUSE_NODE_SLAB 1
44
45 #ifndef MAP_ANONYMOUS
46 #undef FUSE_NODE_SLAB
47 #endif
48
49 #ifndef RENAME_EXCHANGE
50 #define RENAME_EXCHANGE (1 << 1) /* Exchange source and dest */
51 #endif
52
53 #define FUSE_DEFAULT_INTR_SIGNAL SIGUSR1
54
55 #define FUSE_UNKNOWN_INO 0xffffffff
56 #define OFFSET_MAX 0x7fffffffffffffffLL
57
58 #define NODE_TABLE_MIN_SIZE 8192
59
60 struct fuse_fs {
61 struct fuse_operations op;
62 struct fuse_module *m;
63 void *user_data;
64 int debug;
65 };
66
67 struct fusemod_so {
68 void *handle;
69 int ctr;
70 };
71
72 struct lock_queue_element {
73 struct lock_queue_element *next;
74 pthread_cond_t cond;
75 fuse_ino_t nodeid1;
76 const char *name1;
77 char **path1;
78 struct node **wnode1;
79 fuse_ino_t nodeid2;
80 const char *name2;
81 char **path2;
82 struct node **wnode2;
83 int err;
84 bool first_locked : 1;
85 bool second_locked : 1;
86 bool done : 1;
87 };
88
89 struct node_table {
90 struct node **array;
91 size_t use;
92 size_t size;
93 size_t split;
94 };
95
96 #define container_of(ptr, type, member) ({ \
97 const typeof( ((type *)0)->member ) *__mptr = (ptr); \
98 (type *)( (char *)__mptr - offsetof(type,member) );})
99
100 #define list_entry(ptr, type, member) \
101 container_of(ptr, type, member)
102
103 struct list_head {
104 struct list_head *next;
105 struct list_head *prev;
106 };
107
108 struct node_slab {
109 struct list_head list; /* must be the first member */
110 struct list_head freelist;
111 int used;
112 };
113
114 struct fuse {
115 struct fuse_session *se;
116 struct node_table name_table;
117 struct node_table id_table;
118 struct list_head lru_table;
119 fuse_ino_t ctr;
120 unsigned int generation;
121 unsigned int hidectr;
122 pthread_mutex_t lock;
123 struct fuse_config conf;
124 int intr_installed;
125 struct fuse_fs *fs;
126 struct lock_queue_element *lockq;
127 int pagesize;
128 struct list_head partial_slabs;
129 struct list_head full_slabs;
130 pthread_t prune_thread;
131 };
132
133 struct lock {
134 int type;
135 off_t start;
136 off_t end;
137 pid_t pid;
138 uint64_t owner;
139 struct lock *next;
140 };
141
142 struct node {
143 struct node *name_next;
144 struct node *id_next;
145 fuse_ino_t nodeid;
146 unsigned int generation;
147 int refctr;
148 struct node *parent;
149 char *name;
150 uint64_t nlookup;
151 int open_count;
152 struct timespec stat_updated;
153 struct timespec mtime;
154 off_t size;
155 struct lock *locks;
156 unsigned int is_hidden : 1;
157 unsigned int cache_valid : 1;
158 int treelock;
159 char inline_name[32];
160 };
161
162 #define TREELOCK_WRITE -1
163 #define TREELOCK_WAIT_OFFSET INT_MIN
164
165 struct node_lru {
166 struct node node;
167 struct list_head lru;
168 struct timespec forget_time;
169 };
170
171 struct fuse_direntry {
172 struct stat stat;
173 char *name;
174 struct fuse_direntry *next;
175 };
176
177 struct fuse_dh {
178 pthread_mutex_t lock;
179 struct fuse *fuse;
180 fuse_req_t req;
181 char *contents;
182 struct fuse_direntry *first;
183 struct fuse_direntry **last;
184 unsigned len;
185 unsigned size;
186 unsigned needlen;
187 int filled;
188 uint64_t fh;
189 int error;
190 fuse_ino_t nodeid;
191 };
192
193 struct fuse_context_i {
194 struct fuse_context ctx;
195 fuse_req_t req;
196 };
197
198 /* Defined by FUSE_REGISTER_MODULE() in lib/modules/subdir.c and iconv.c. */
199 extern fuse_module_factory_t fuse_module_subdir_factory;
200 #ifdef HAVE_ICONV
201 extern fuse_module_factory_t fuse_module_iconv_factory;
202 #endif
203
204 static pthread_key_t fuse_context_key;
205 static pthread_mutex_t fuse_context_lock = PTHREAD_MUTEX_INITIALIZER;
206 static int fuse_context_ref;
207 static struct fuse_module *fuse_modules = NULL;
208
fuse_register_module(const char * name,fuse_module_factory_t factory,struct fusemod_so * so)209 static int fuse_register_module(const char *name,
210 fuse_module_factory_t factory,
211 struct fusemod_so *so)
212 {
213 struct fuse_module *mod;
214
215 mod = calloc(1, sizeof(struct fuse_module));
216 if (!mod) {
217 fuse_log(FUSE_LOG_ERR, "fuse: failed to allocate module\n");
218 return -1;
219 }
220 mod->name = strdup(name);
221 if (!mod->name) {
222 fuse_log(FUSE_LOG_ERR, "fuse: failed to allocate module name\n");
223 free(mod);
224 return -1;
225 }
226 mod->factory = factory;
227 mod->ctr = 0;
228 mod->so = so;
229 if (mod->so)
230 mod->so->ctr++;
231 mod->next = fuse_modules;
232 fuse_modules = mod;
233
234 return 0;
235 }
236
fuse_unregister_module(struct fuse_module * m)237 static void fuse_unregister_module(struct fuse_module *m)
238 {
239 struct fuse_module **mp;
240 for (mp = &fuse_modules; *mp; mp = &(*mp)->next) {
241 if (*mp == m) {
242 *mp = (*mp)->next;
243 break;
244 }
245 }
246 free(m->name);
247 free(m);
248 }
249
fuse_load_so_module(const char * module)250 static int fuse_load_so_module(const char *module)
251 {
252 int ret = -1;
253 char *tmp;
254 struct fusemod_so *so;
255 fuse_module_factory_t factory;
256
257 tmp = malloc(strlen(module) + 64);
258 if (!tmp) {
259 fuse_log(FUSE_LOG_ERR, "fuse: memory allocation failed\n");
260 return -1;
261 }
262 sprintf(tmp, "libfusemod_%s.so", module);
263 so = calloc(1, sizeof(struct fusemod_so));
264 if (!so) {
265 fuse_log(FUSE_LOG_ERR, "fuse: failed to allocate module so\n");
266 goto out;
267 }
268
269 so->handle = dlopen(tmp, RTLD_NOW);
270 if (so->handle == NULL) {
271 fuse_log(FUSE_LOG_ERR, "fuse: dlopen(%s) failed: %s\n",
272 tmp, dlerror());
273 goto out_free_so;
274 }
275
276 sprintf(tmp, "fuse_module_%s_factory", module);
277 *(void**)(&factory) = dlsym(so->handle, tmp);
278 if (factory == NULL) {
279 fuse_log(FUSE_LOG_ERR, "fuse: symbol <%s> not found in module: %s\n",
280 tmp, dlerror());
281 goto out_dlclose;
282 }
283 ret = fuse_register_module(module, factory, so);
284 if (ret)
285 goto out_dlclose;
286
287 out:
288 free(tmp);
289 return ret;
290
291 out_dlclose:
292 dlclose(so->handle);
293 out_free_so:
294 free(so);
295 goto out;
296 }
297
fuse_find_module(const char * module)298 static struct fuse_module *fuse_find_module(const char *module)
299 {
300 struct fuse_module *m;
301 for (m = fuse_modules; m; m = m->next) {
302 if (strcmp(module, m->name) == 0) {
303 m->ctr++;
304 break;
305 }
306 }
307 return m;
308 }
309
fuse_get_module(const char * module)310 static struct fuse_module *fuse_get_module(const char *module)
311 {
312 struct fuse_module *m;
313
314 pthread_mutex_lock(&fuse_context_lock);
315 m = fuse_find_module(module);
316 if (!m) {
317 int err = fuse_load_so_module(module);
318 if (!err)
319 m = fuse_find_module(module);
320 }
321 pthread_mutex_unlock(&fuse_context_lock);
322 return m;
323 }
324
fuse_put_module(struct fuse_module * m)325 static void fuse_put_module(struct fuse_module *m)
326 {
327 pthread_mutex_lock(&fuse_context_lock);
328 if (m->so)
329 assert(m->ctr > 0);
330 /* Builtin modules may already have m->ctr == 0 */
331 if (m->ctr > 0)
332 m->ctr--;
333 if (!m->ctr && m->so) {
334 struct fusemod_so *so = m->so;
335 assert(so->ctr > 0);
336 so->ctr--;
337 if (!so->ctr) {
338 struct fuse_module **mp;
339 for (mp = &fuse_modules; *mp;) {
340 if ((*mp)->so == so)
341 fuse_unregister_module(*mp);
342 else
343 mp = &(*mp)->next;
344 }
345 dlclose(so->handle);
346 free(so);
347 }
348 } else if (!m->ctr) {
349 fuse_unregister_module(m);
350 }
351 pthread_mutex_unlock(&fuse_context_lock);
352 }
353
init_list_head(struct list_head * list)354 static void init_list_head(struct list_head *list)
355 {
356 list->next = list;
357 list->prev = list;
358 }
359
list_empty(const struct list_head * head)360 static int list_empty(const struct list_head *head)
361 {
362 return head->next == head;
363 }
364
list_add(struct list_head * new,struct list_head * prev,struct list_head * next)365 static void list_add(struct list_head *new, struct list_head *prev,
366 struct list_head *next)
367 {
368 next->prev = new;
369 new->next = next;
370 new->prev = prev;
371 prev->next = new;
372 }
373
list_add_head(struct list_head * new,struct list_head * head)374 static inline void list_add_head(struct list_head *new, struct list_head *head)
375 {
376 list_add(new, head, head->next);
377 }
378
list_add_tail(struct list_head * new,struct list_head * head)379 static inline void list_add_tail(struct list_head *new, struct list_head *head)
380 {
381 list_add(new, head->prev, head);
382 }
383
list_del(struct list_head * entry)384 static inline void list_del(struct list_head *entry)
385 {
386 struct list_head *prev = entry->prev;
387 struct list_head *next = entry->next;
388
389 next->prev = prev;
390 prev->next = next;
391 }
392
lru_enabled(struct fuse * f)393 static inline int lru_enabled(struct fuse *f)
394 {
395 return f->conf.remember > 0;
396 }
397
node_lru(struct node * node)398 static struct node_lru *node_lru(struct node *node)
399 {
400 return (struct node_lru *) node;
401 }
402
get_node_size(struct fuse * f)403 static size_t get_node_size(struct fuse *f)
404 {
405 if (lru_enabled(f))
406 return sizeof(struct node_lru);
407 else
408 return sizeof(struct node);
409 }
410
411 #ifdef FUSE_NODE_SLAB
list_to_slab(struct list_head * head)412 static struct node_slab *list_to_slab(struct list_head *head)
413 {
414 return (struct node_slab *) head;
415 }
416
node_to_slab(struct fuse * f,struct node * node)417 static struct node_slab *node_to_slab(struct fuse *f, struct node *node)
418 {
419 return (struct node_slab *) (((uintptr_t) node) & ~((uintptr_t) f->pagesize - 1));
420 }
421
alloc_slab(struct fuse * f)422 static int alloc_slab(struct fuse *f)
423 {
424 void *mem;
425 struct node_slab *slab;
426 char *start;
427 size_t num;
428 size_t i;
429 size_t node_size = get_node_size(f);
430
431 mem = mmap(NULL, f->pagesize, PROT_READ | PROT_WRITE,
432 MAP_PRIVATE | MAP_ANONYMOUS, -1, 0);
433
434 if (mem == MAP_FAILED)
435 return -1;
436
437 slab = mem;
438 init_list_head(&slab->freelist);
439 slab->used = 0;
440 num = (f->pagesize - sizeof(struct node_slab)) / node_size;
441
442 start = (char *) mem + f->pagesize - num * node_size;
443 for (i = 0; i < num; i++) {
444 struct list_head *n;
445
446 n = (struct list_head *) (start + i * node_size);
447 list_add_tail(n, &slab->freelist);
448 }
449 list_add_tail(&slab->list, &f->partial_slabs);
450
451 return 0;
452 }
453
alloc_node(struct fuse * f)454 static struct node *alloc_node(struct fuse *f)
455 {
456 struct node_slab *slab;
457 struct list_head *node;
458
459 if (list_empty(&f->partial_slabs)) {
460 int res = alloc_slab(f);
461 if (res != 0)
462 return NULL;
463 }
464 slab = list_to_slab(f->partial_slabs.next);
465 slab->used++;
466 node = slab->freelist.next;
467 list_del(node);
468 if (list_empty(&slab->freelist)) {
469 list_del(&slab->list);
470 list_add_tail(&slab->list, &f->full_slabs);
471 }
472 memset(node, 0, sizeof(struct node));
473
474 return (struct node *) node;
475 }
476
free_slab(struct fuse * f,struct node_slab * slab)477 static void free_slab(struct fuse *f, struct node_slab *slab)
478 {
479 int res;
480
481 list_del(&slab->list);
482 res = munmap(slab, f->pagesize);
483 if (res == -1)
484 fuse_log(FUSE_LOG_WARNING, "fuse warning: munmap(%p) failed\n",
485 slab);
486 }
487
free_node_mem(struct fuse * f,struct node * node)488 static void free_node_mem(struct fuse *f, struct node *node)
489 {
490 struct node_slab *slab = node_to_slab(f, node);
491 struct list_head *n = (struct list_head *) node;
492
493 slab->used--;
494 if (slab->used) {
495 if (list_empty(&slab->freelist)) {
496 list_del(&slab->list);
497 list_add_tail(&slab->list, &f->partial_slabs);
498 }
499 list_add_head(n, &slab->freelist);
500 } else {
501 free_slab(f, slab);
502 }
503 }
504 #else
alloc_node(struct fuse * f)505 static struct node *alloc_node(struct fuse *f)
506 {
507 return (struct node *) calloc(1, get_node_size(f));
508 }
509
free_node_mem(struct fuse * f,struct node * node)510 static void free_node_mem(struct fuse *f, struct node *node)
511 {
512 (void) f;
513 free(node);
514 }
515 #endif
516
id_hash(struct fuse * f,fuse_ino_t ino)517 static size_t id_hash(struct fuse *f, fuse_ino_t ino)
518 {
519 uint64_t hash = ((uint32_t) ino * 2654435761U) % f->id_table.size;
520 uint64_t oldhash = hash % (f->id_table.size / 2);
521
522 if (oldhash >= f->id_table.split)
523 return oldhash;
524 else
525 return hash;
526 }
527
get_node_nocheck(struct fuse * f,fuse_ino_t nodeid)528 static struct node *get_node_nocheck(struct fuse *f, fuse_ino_t nodeid)
529 {
530 size_t hash = id_hash(f, nodeid);
531 struct node *node;
532
533 for (node = f->id_table.array[hash]; node != NULL; node = node->id_next)
534 if (node->nodeid == nodeid)
535 return node;
536
537 return NULL;
538 }
539
get_node(struct fuse * f,fuse_ino_t nodeid)540 static struct node *get_node(struct fuse *f, fuse_ino_t nodeid)
541 {
542 struct node *node = get_node_nocheck(f, nodeid);
543 if (!node) {
544 fuse_log(FUSE_LOG_ERR, "fuse internal error: node %llu not found\n",
545 (unsigned long long) nodeid);
546 abort();
547 }
548 return node;
549 }
550
551 static void curr_time(struct timespec *now);
552 static double diff_timespec(const struct timespec *t1,
553 const struct timespec *t2);
554
remove_node_lru(struct node * node)555 static void remove_node_lru(struct node *node)
556 {
557 struct node_lru *lnode = node_lru(node);
558 list_del(&lnode->lru);
559 init_list_head(&lnode->lru);
560 }
561
set_forget_time(struct fuse * f,struct node * node)562 static void set_forget_time(struct fuse *f, struct node *node)
563 {
564 struct node_lru *lnode = node_lru(node);
565
566 list_del(&lnode->lru);
567 list_add_tail(&lnode->lru, &f->lru_table);
568 curr_time(&lnode->forget_time);
569 }
570
free_node(struct fuse * f,struct node * node)571 static void free_node(struct fuse *f, struct node *node)
572 {
573 if (node->name != node->inline_name)
574 free(node->name);
575 free_node_mem(f, node);
576 }
577
node_table_reduce(struct node_table * t)578 static void node_table_reduce(struct node_table *t)
579 {
580 size_t newsize = t->size / 2;
581 void *newarray;
582
583 if (newsize < NODE_TABLE_MIN_SIZE)
584 return;
585
586 newarray = realloc(t->array, sizeof(struct node *) * newsize);
587 if (newarray != NULL)
588 t->array = newarray;
589
590 t->size = newsize;
591 t->split = t->size / 2;
592 }
593
remerge_id(struct fuse * f)594 static void remerge_id(struct fuse *f)
595 {
596 struct node_table *t = &f->id_table;
597 int iter;
598
599 if (t->split == 0)
600 node_table_reduce(t);
601
602 for (iter = 8; t->split > 0 && iter; iter--) {
603 struct node **upper;
604
605 t->split--;
606 upper = &t->array[t->split + t->size / 2];
607 if (*upper) {
608 struct node **nodep;
609
610 for (nodep = &t->array[t->split]; *nodep;
611 nodep = &(*nodep)->id_next);
612
613 *nodep = *upper;
614 *upper = NULL;
615 break;
616 }
617 }
618 }
619
unhash_id(struct fuse * f,struct node * node)620 static void unhash_id(struct fuse *f, struct node *node)
621 {
622 struct node **nodep = &f->id_table.array[id_hash(f, node->nodeid)];
623
624 for (; *nodep != NULL; nodep = &(*nodep)->id_next)
625 if (*nodep == node) {
626 *nodep = node->id_next;
627 f->id_table.use--;
628
629 if(f->id_table.use < f->id_table.size / 4)
630 remerge_id(f);
631 return;
632 }
633 }
634
node_table_resize(struct node_table * t)635 static int node_table_resize(struct node_table *t)
636 {
637 size_t newsize = t->size * 2;
638 void *newarray;
639
640 newarray = realloc(t->array, sizeof(struct node *) * newsize);
641 if (newarray == NULL)
642 return -1;
643
644 t->array = newarray;
645 memset(t->array + t->size, 0, t->size * sizeof(struct node *));
646 t->size = newsize;
647 t->split = 0;
648
649 return 0;
650 }
651
rehash_id(struct fuse * f)652 static void rehash_id(struct fuse *f)
653 {
654 struct node_table *t = &f->id_table;
655 struct node **nodep;
656 struct node **next;
657 size_t hash;
658
659 if (t->split == t->size / 2)
660 return;
661
662 hash = t->split;
663 t->split++;
664 for (nodep = &t->array[hash]; *nodep != NULL; nodep = next) {
665 struct node *node = *nodep;
666 size_t newhash = id_hash(f, node->nodeid);
667
668 if (newhash != hash) {
669 next = nodep;
670 *nodep = node->id_next;
671 node->id_next = t->array[newhash];
672 t->array[newhash] = node;
673 } else {
674 next = &node->id_next;
675 }
676 }
677 if (t->split == t->size / 2)
678 node_table_resize(t);
679 }
680
hash_id(struct fuse * f,struct node * node)681 static void hash_id(struct fuse *f, struct node *node)
682 {
683 size_t hash = id_hash(f, node->nodeid);
684 node->id_next = f->id_table.array[hash];
685 f->id_table.array[hash] = node;
686 f->id_table.use++;
687
688 if (f->id_table.use >= f->id_table.size / 2)
689 rehash_id(f);
690 }
691
name_hash(struct fuse * f,fuse_ino_t parent,const char * name)692 static size_t name_hash(struct fuse *f, fuse_ino_t parent,
693 const char *name)
694 {
695 uint64_t hash = parent;
696 uint64_t oldhash;
697
698 for (; *name; name++)
699 hash = hash * 31 + (unsigned char) *name;
700
701 hash %= f->name_table.size;
702 oldhash = hash % (f->name_table.size / 2);
703 if (oldhash >= f->name_table.split)
704 return oldhash;
705 else
706 return hash;
707 }
708
709 static void unref_node(struct fuse *f, struct node *node);
710
remerge_name(struct fuse * f)711 static void remerge_name(struct fuse *f)
712 {
713 struct node_table *t = &f->name_table;
714 int iter;
715
716 if (t->split == 0)
717 node_table_reduce(t);
718
719 for (iter = 8; t->split > 0 && iter; iter--) {
720 struct node **upper;
721
722 t->split--;
723 upper = &t->array[t->split + t->size / 2];
724 if (*upper) {
725 struct node **nodep;
726
727 for (nodep = &t->array[t->split]; *nodep;
728 nodep = &(*nodep)->name_next);
729
730 *nodep = *upper;
731 *upper = NULL;
732 break;
733 }
734 }
735 }
736
unhash_name(struct fuse * f,struct node * node)737 static void unhash_name(struct fuse *f, struct node *node)
738 {
739 if (node->name) {
740 size_t hash = name_hash(f, node->parent->nodeid, node->name);
741 struct node **nodep = &f->name_table.array[hash];
742
743 for (; *nodep != NULL; nodep = &(*nodep)->name_next)
744 if (*nodep == node) {
745 *nodep = node->name_next;
746 node->name_next = NULL;
747 unref_node(f, node->parent);
748 if (node->name != node->inline_name)
749 free(node->name);
750 node->name = NULL;
751 node->parent = NULL;
752 f->name_table.use--;
753
754 if (f->name_table.use < f->name_table.size / 4)
755 remerge_name(f);
756 return;
757 }
758 fuse_log(FUSE_LOG_ERR,
759 "fuse internal error: unable to unhash node: %llu\n",
760 (unsigned long long) node->nodeid);
761 abort();
762 }
763 }
764
rehash_name(struct fuse * f)765 static void rehash_name(struct fuse *f)
766 {
767 struct node_table *t = &f->name_table;
768 struct node **nodep;
769 struct node **next;
770 size_t hash;
771
772 if (t->split == t->size / 2)
773 return;
774
775 hash = t->split;
776 t->split++;
777 for (nodep = &t->array[hash]; *nodep != NULL; nodep = next) {
778 struct node *node = *nodep;
779 size_t newhash = name_hash(f, node->parent->nodeid, node->name);
780
781 if (newhash != hash) {
782 next = nodep;
783 *nodep = node->name_next;
784 node->name_next = t->array[newhash];
785 t->array[newhash] = node;
786 } else {
787 next = &node->name_next;
788 }
789 }
790 if (t->split == t->size / 2)
791 node_table_resize(t);
792 }
793
hash_name(struct fuse * f,struct node * node,fuse_ino_t parentid,const char * name)794 static int hash_name(struct fuse *f, struct node *node, fuse_ino_t parentid,
795 const char *name)
796 {
797 size_t hash = name_hash(f, parentid, name);
798 struct node *parent = get_node(f, parentid);
799 if (strlen(name) < sizeof(node->inline_name)) {
800 strcpy(node->inline_name, name);
801 node->name = node->inline_name;
802 } else {
803 node->name = strdup(name);
804 if (node->name == NULL)
805 return -1;
806 }
807
808 parent->refctr ++;
809 node->parent = parent;
810 node->name_next = f->name_table.array[hash];
811 f->name_table.array[hash] = node;
812 f->name_table.use++;
813
814 if (f->name_table.use >= f->name_table.size / 2)
815 rehash_name(f);
816
817 return 0;
818 }
819
delete_node(struct fuse * f,struct node * node)820 static void delete_node(struct fuse *f, struct node *node)
821 {
822 if (f->conf.debug)
823 fuse_log(FUSE_LOG_DEBUG, "DELETE: %llu\n",
824 (unsigned long long) node->nodeid);
825
826 assert(node->treelock == 0);
827 unhash_name(f, node);
828 if (lru_enabled(f))
829 remove_node_lru(node);
830 unhash_id(f, node);
831 free_node(f, node);
832 }
833
unref_node(struct fuse * f,struct node * node)834 static void unref_node(struct fuse *f, struct node *node)
835 {
836 assert(node->refctr > 0);
837 node->refctr --;
838 if (!node->refctr)
839 delete_node(f, node);
840 }
841
next_id(struct fuse * f)842 static fuse_ino_t next_id(struct fuse *f)
843 {
844 do {
845 f->ctr = (f->ctr + 1) & 0xffffffff;
846 if (!f->ctr)
847 f->generation ++;
848 } while (f->ctr == 0 || f->ctr == FUSE_UNKNOWN_INO ||
849 get_node_nocheck(f, f->ctr) != NULL);
850 return f->ctr;
851 }
852
lookup_node(struct fuse * f,fuse_ino_t parent,const char * name)853 static struct node *lookup_node(struct fuse *f, fuse_ino_t parent,
854 const char *name)
855 {
856 size_t hash = name_hash(f, parent, name);
857 struct node *node;
858
859 for (node = f->name_table.array[hash]; node != NULL; node = node->name_next)
860 if (node->parent->nodeid == parent &&
861 strcmp(node->name, name) == 0)
862 return node;
863
864 return NULL;
865 }
866
inc_nlookup(struct node * node)867 static void inc_nlookup(struct node *node)
868 {
869 if (!node->nlookup)
870 node->refctr++;
871 node->nlookup++;
872 }
873
find_node(struct fuse * f,fuse_ino_t parent,const char * name)874 static struct node *find_node(struct fuse *f, fuse_ino_t parent,
875 const char *name)
876 {
877 struct node *node;
878
879 pthread_mutex_lock(&f->lock);
880 if (!name)
881 node = get_node(f, parent);
882 else
883 node = lookup_node(f, parent, name);
884 if (node == NULL) {
885 node = alloc_node(f);
886 if (node == NULL)
887 goto out_err;
888
889 node->nodeid = next_id(f);
890 node->generation = f->generation;
891 if (f->conf.remember)
892 inc_nlookup(node);
893
894 if (hash_name(f, node, parent, name) == -1) {
895 free_node(f, node);
896 node = NULL;
897 goto out_err;
898 }
899 hash_id(f, node);
900 if (lru_enabled(f)) {
901 struct node_lru *lnode = node_lru(node);
902 init_list_head(&lnode->lru);
903 }
904 } else if (lru_enabled(f) && node->nlookup == 1) {
905 remove_node_lru(node);
906 }
907 inc_nlookup(node);
908 out_err:
909 pthread_mutex_unlock(&f->lock);
910 return node;
911 }
912
lookup_path_in_cache(struct fuse * f,const char * path,fuse_ino_t * inop)913 static int lookup_path_in_cache(struct fuse *f,
914 const char *path, fuse_ino_t *inop)
915 {
916 char *tmp = strdup(path);
917 if (!tmp)
918 return -ENOMEM;
919
920 pthread_mutex_lock(&f->lock);
921 fuse_ino_t ino = FUSE_ROOT_ID;
922
923 int err = 0;
924 char *save_ptr;
925 char *path_element = strtok_r(tmp, "/", &save_ptr);
926 while (path_element != NULL) {
927 struct node *node = lookup_node(f, ino, path_element);
928 if (node == NULL) {
929 err = -ENOENT;
930 break;
931 }
932 ino = node->nodeid;
933 path_element = strtok_r(NULL, "/", &save_ptr);
934 }
935 pthread_mutex_unlock(&f->lock);
936 free(tmp);
937
938 if (!err)
939 *inop = ino;
940 return err;
941 }
942
add_name(char ** buf,unsigned * bufsize,char * s,const char * name)943 static char *add_name(char **buf, unsigned *bufsize, char *s, const char *name)
944 {
945 size_t len = strlen(name);
946
947 if (s - len <= *buf) {
948 unsigned pathlen = *bufsize - (s - *buf);
949 unsigned newbufsize = *bufsize;
950 char *newbuf;
951
952 while (newbufsize < pathlen + len + 1) {
953 if (newbufsize >= 0x80000000)
954 newbufsize = 0xffffffff;
955 else
956 newbufsize *= 2;
957 }
958
959 newbuf = realloc(*buf, newbufsize);
960 if (newbuf == NULL)
961 return NULL;
962
963 *buf = newbuf;
964 s = newbuf + newbufsize - pathlen;
965 memmove(s, newbuf + *bufsize - pathlen, pathlen);
966 *bufsize = newbufsize;
967 }
968 s -= len;
969 memcpy(s, name, len);
970 s--;
971 *s = '/';
972
973 return s;
974 }
975
unlock_path(struct fuse * f,fuse_ino_t nodeid,struct node * wnode,struct node * end)976 static void unlock_path(struct fuse *f, fuse_ino_t nodeid, struct node *wnode,
977 struct node *end)
978 {
979 struct node *node;
980
981 if (wnode) {
982 assert(wnode->treelock == TREELOCK_WRITE);
983 wnode->treelock = 0;
984 }
985
986 for (node = get_node(f, nodeid);
987 node != end && node->nodeid != FUSE_ROOT_ID; node = node->parent) {
988 assert(node->treelock != 0);
989 assert(node->treelock != TREELOCK_WAIT_OFFSET);
990 assert(node->treelock != TREELOCK_WRITE);
991 node->treelock--;
992 if (node->treelock == TREELOCK_WAIT_OFFSET)
993 node->treelock = 0;
994 }
995 }
996
try_get_path(struct fuse * f,fuse_ino_t nodeid,const char * name,char ** path,struct node ** wnodep,bool need_lock)997 static int try_get_path(struct fuse *f, fuse_ino_t nodeid, const char *name,
998 char **path, struct node **wnodep, bool need_lock)
999 {
1000 unsigned bufsize = 256;
1001 char *buf;
1002 char *s;
1003 struct node *node;
1004 struct node *wnode = NULL;
1005 int err;
1006
1007 *path = NULL;
1008
1009 err = -ENOMEM;
1010 buf = malloc(bufsize);
1011 if (buf == NULL)
1012 goto out_err;
1013
1014 s = buf + bufsize - 1;
1015 *s = '\0';
1016
1017 if (name != NULL) {
1018 s = add_name(&buf, &bufsize, s, name);
1019 err = -ENOMEM;
1020 if (s == NULL)
1021 goto out_free;
1022 }
1023
1024 if (wnodep) {
1025 assert(need_lock);
1026 wnode = lookup_node(f, nodeid, name);
1027 if (wnode) {
1028 if (wnode->treelock != 0) {
1029 if (wnode->treelock > 0)
1030 wnode->treelock += TREELOCK_WAIT_OFFSET;
1031 err = -EAGAIN;
1032 goto out_free;
1033 }
1034 wnode->treelock = TREELOCK_WRITE;
1035 }
1036 }
1037
1038 for (node = get_node(f, nodeid); node->nodeid != FUSE_ROOT_ID;
1039 node = node->parent) {
1040 err = -ENOENT;
1041 if (node->name == NULL || node->parent == NULL)
1042 goto out_unlock;
1043
1044 err = -ENOMEM;
1045 s = add_name(&buf, &bufsize, s, node->name);
1046 if (s == NULL)
1047 goto out_unlock;
1048
1049 if (need_lock) {
1050 err = -EAGAIN;
1051 if (node->treelock < 0)
1052 goto out_unlock;
1053
1054 node->treelock++;
1055 }
1056 }
1057
1058 if (s[0])
1059 memmove(buf, s, bufsize - (s - buf));
1060 else
1061 strcpy(buf, "/");
1062
1063 *path = buf;
1064 if (wnodep)
1065 *wnodep = wnode;
1066
1067 return 0;
1068
1069 out_unlock:
1070 if (need_lock)
1071 unlock_path(f, nodeid, wnode, node);
1072 out_free:
1073 free(buf);
1074
1075 out_err:
1076 return err;
1077 }
1078
queue_element_unlock(struct fuse * f,struct lock_queue_element * qe)1079 static void queue_element_unlock(struct fuse *f, struct lock_queue_element *qe)
1080 {
1081 struct node *wnode;
1082
1083 if (qe->first_locked) {
1084 wnode = qe->wnode1 ? *qe->wnode1 : NULL;
1085 unlock_path(f, qe->nodeid1, wnode, NULL);
1086 qe->first_locked = false;
1087 }
1088 if (qe->second_locked) {
1089 wnode = qe->wnode2 ? *qe->wnode2 : NULL;
1090 unlock_path(f, qe->nodeid2, wnode, NULL);
1091 qe->second_locked = false;
1092 }
1093 }
1094
queue_element_wakeup(struct fuse * f,struct lock_queue_element * qe)1095 static void queue_element_wakeup(struct fuse *f, struct lock_queue_element *qe)
1096 {
1097 int err;
1098 bool first = (qe == f->lockq);
1099
1100 if (!qe->path1) {
1101 /* Just waiting for it to be unlocked */
1102 if (get_node(f, qe->nodeid1)->treelock == 0)
1103 pthread_cond_signal(&qe->cond);
1104
1105 return;
1106 }
1107
1108 if (!qe->first_locked) {
1109 err = try_get_path(f, qe->nodeid1, qe->name1, qe->path1,
1110 qe->wnode1, true);
1111 if (!err)
1112 qe->first_locked = true;
1113 else if (err != -EAGAIN)
1114 goto err_unlock;
1115 }
1116 if (!qe->second_locked && qe->path2) {
1117 err = try_get_path(f, qe->nodeid2, qe->name2, qe->path2,
1118 qe->wnode2, true);
1119 if (!err)
1120 qe->second_locked = true;
1121 else if (err != -EAGAIN)
1122 goto err_unlock;
1123 }
1124
1125 if (qe->first_locked && (qe->second_locked || !qe->path2)) {
1126 err = 0;
1127 goto done;
1128 }
1129
1130 /*
1131 * Only let the first element be partially locked otherwise there could
1132 * be a deadlock.
1133 *
1134 * But do allow the first element to be partially locked to prevent
1135 * starvation.
1136 */
1137 if (!first)
1138 queue_element_unlock(f, qe);
1139
1140 /* keep trying */
1141 return;
1142
1143 err_unlock:
1144 queue_element_unlock(f, qe);
1145 done:
1146 qe->err = err;
1147 qe->done = true;
1148 pthread_cond_signal(&qe->cond);
1149 }
1150
wake_up_queued(struct fuse * f)1151 static void wake_up_queued(struct fuse *f)
1152 {
1153 struct lock_queue_element *qe;
1154
1155 for (qe = f->lockq; qe != NULL; qe = qe->next)
1156 queue_element_wakeup(f, qe);
1157 }
1158
debug_path(struct fuse * f,const char * msg,fuse_ino_t nodeid,const char * name,bool wr)1159 static void debug_path(struct fuse *f, const char *msg, fuse_ino_t nodeid,
1160 const char *name, bool wr)
1161 {
1162 if (f->conf.debug) {
1163 struct node *wnode = NULL;
1164
1165 if (wr)
1166 wnode = lookup_node(f, nodeid, name);
1167
1168 if (wnode) {
1169 fuse_log(FUSE_LOG_DEBUG, "%s %llu (w)\n",
1170 msg, (unsigned long long) wnode->nodeid);
1171 } else {
1172 fuse_log(FUSE_LOG_DEBUG, "%s %llu\n",
1173 msg, (unsigned long long) nodeid);
1174 }
1175 }
1176 }
1177
queue_path(struct fuse * f,struct lock_queue_element * qe)1178 static void queue_path(struct fuse *f, struct lock_queue_element *qe)
1179 {
1180 struct lock_queue_element **qp;
1181
1182 qe->done = false;
1183 qe->first_locked = false;
1184 qe->second_locked = false;
1185 pthread_cond_init(&qe->cond, NULL);
1186 qe->next = NULL;
1187 for (qp = &f->lockq; *qp != NULL; qp = &(*qp)->next);
1188 *qp = qe;
1189 }
1190
dequeue_path(struct fuse * f,struct lock_queue_element * qe)1191 static void dequeue_path(struct fuse *f, struct lock_queue_element *qe)
1192 {
1193 struct lock_queue_element **qp;
1194
1195 pthread_cond_destroy(&qe->cond);
1196 for (qp = &f->lockq; *qp != qe; qp = &(*qp)->next);
1197 *qp = qe->next;
1198 }
1199
wait_path(struct fuse * f,struct lock_queue_element * qe)1200 static int wait_path(struct fuse *f, struct lock_queue_element *qe)
1201 {
1202 queue_path(f, qe);
1203
1204 do {
1205 pthread_cond_wait(&qe->cond, &f->lock);
1206 } while (!qe->done);
1207
1208 dequeue_path(f, qe);
1209
1210 return qe->err;
1211 }
1212
get_path_common(struct fuse * f,fuse_ino_t nodeid,const char * name,char ** path,struct node ** wnode)1213 static int get_path_common(struct fuse *f, fuse_ino_t nodeid, const char *name,
1214 char **path, struct node **wnode)
1215 {
1216 int err;
1217
1218 pthread_mutex_lock(&f->lock);
1219 err = try_get_path(f, nodeid, name, path, wnode, true);
1220 if (err == -EAGAIN) {
1221 struct lock_queue_element qe = {
1222 .nodeid1 = nodeid,
1223 .name1 = name,
1224 .path1 = path,
1225 .wnode1 = wnode,
1226 };
1227 debug_path(f, "QUEUE PATH", nodeid, name, !!wnode);
1228 err = wait_path(f, &qe);
1229 debug_path(f, "DEQUEUE PATH", nodeid, name, !!wnode);
1230 }
1231 pthread_mutex_unlock(&f->lock);
1232
1233 return err;
1234 }
1235
get_path(struct fuse * f,fuse_ino_t nodeid,char ** path)1236 static int get_path(struct fuse *f, fuse_ino_t nodeid, char **path)
1237 {
1238 return get_path_common(f, nodeid, NULL, path, NULL);
1239 }
1240
get_path_nullok(struct fuse * f,fuse_ino_t nodeid,char ** path)1241 static int get_path_nullok(struct fuse *f, fuse_ino_t nodeid, char **path)
1242 {
1243 int err = 0;
1244
1245 if (f->conf.nullpath_ok) {
1246 *path = NULL;
1247 } else {
1248 err = get_path_common(f, nodeid, NULL, path, NULL);
1249 if (err == -ENOENT)
1250 err = 0;
1251 }
1252
1253 return err;
1254 }
1255
get_path_name(struct fuse * f,fuse_ino_t nodeid,const char * name,char ** path)1256 static int get_path_name(struct fuse *f, fuse_ino_t nodeid, const char *name,
1257 char **path)
1258 {
1259 return get_path_common(f, nodeid, name, path, NULL);
1260 }
1261
get_path_wrlock(struct fuse * f,fuse_ino_t nodeid,const char * name,char ** path,struct node ** wnode)1262 static int get_path_wrlock(struct fuse *f, fuse_ino_t nodeid, const char *name,
1263 char **path, struct node **wnode)
1264 {
1265 return get_path_common(f, nodeid, name, path, wnode);
1266 }
1267
1268 #if defined(__FreeBSD__)
1269 #define CHECK_DIR_LOOP
1270 #endif
1271
1272 #if defined(CHECK_DIR_LOOP)
check_dir_loop(struct fuse * f,fuse_ino_t nodeid1,const char * name1,fuse_ino_t nodeid2,const char * name2)1273 static int check_dir_loop(struct fuse *f,
1274 fuse_ino_t nodeid1, const char *name1,
1275 fuse_ino_t nodeid2, const char *name2)
1276 {
1277 struct node *node, *node1, *node2;
1278 fuse_ino_t id1, id2;
1279
1280 node1 = lookup_node(f, nodeid1, name1);
1281 id1 = node1 ? node1->nodeid : nodeid1;
1282
1283 node2 = lookup_node(f, nodeid2, name2);
1284 id2 = node2 ? node2->nodeid : nodeid2;
1285
1286 for (node = get_node(f, id2); node->nodeid != FUSE_ROOT_ID;
1287 node = node->parent) {
1288 if (node->name == NULL || node->parent == NULL)
1289 break;
1290
1291 if (node->nodeid != id2 && node->nodeid == id1)
1292 return -EINVAL;
1293 }
1294
1295 if (node2)
1296 {
1297 for (node = get_node(f, id1); node->nodeid != FUSE_ROOT_ID;
1298 node = node->parent) {
1299 if (node->name == NULL || node->parent == NULL)
1300 break;
1301
1302 if (node->nodeid != id1 && node->nodeid == id2)
1303 return -ENOTEMPTY;
1304 }
1305 }
1306
1307 return 0;
1308 }
1309 #endif
1310
try_get_path2(struct fuse * f,fuse_ino_t nodeid1,const char * name1,fuse_ino_t nodeid2,const char * name2,char ** path1,char ** path2,struct node ** wnode1,struct node ** wnode2)1311 static int try_get_path2(struct fuse *f, fuse_ino_t nodeid1, const char *name1,
1312 fuse_ino_t nodeid2, const char *name2,
1313 char **path1, char **path2,
1314 struct node **wnode1, struct node **wnode2)
1315 {
1316 int err;
1317
1318 /* FIXME: locking two paths needs deadlock checking */
1319 err = try_get_path(f, nodeid1, name1, path1, wnode1, true);
1320 if (!err) {
1321 err = try_get_path(f, nodeid2, name2, path2, wnode2, true);
1322 if (err) {
1323 struct node *wn1 = wnode1 ? *wnode1 : NULL;
1324
1325 unlock_path(f, nodeid1, wn1, NULL);
1326 free(*path1);
1327 }
1328 }
1329 return err;
1330 }
1331
get_path2(struct fuse * f,fuse_ino_t nodeid1,const char * name1,fuse_ino_t nodeid2,const char * name2,char ** path1,char ** path2,struct node ** wnode1,struct node ** wnode2)1332 static int get_path2(struct fuse *f, fuse_ino_t nodeid1, const char *name1,
1333 fuse_ino_t nodeid2, const char *name2,
1334 char **path1, char **path2,
1335 struct node **wnode1, struct node **wnode2)
1336 {
1337 int err;
1338
1339 pthread_mutex_lock(&f->lock);
1340
1341 #if defined(CHECK_DIR_LOOP)
1342 if (name1)
1343 {
1344 // called during rename; perform dir loop check
1345 err = check_dir_loop(f, nodeid1, name1, nodeid2, name2);
1346 if (err)
1347 goto out_unlock;
1348 }
1349 #endif
1350
1351 err = try_get_path2(f, nodeid1, name1, nodeid2, name2,
1352 path1, path2, wnode1, wnode2);
1353 if (err == -EAGAIN) {
1354 struct lock_queue_element qe = {
1355 .nodeid1 = nodeid1,
1356 .name1 = name1,
1357 .path1 = path1,
1358 .wnode1 = wnode1,
1359 .nodeid2 = nodeid2,
1360 .name2 = name2,
1361 .path2 = path2,
1362 .wnode2 = wnode2,
1363 };
1364
1365 debug_path(f, "QUEUE PATH1", nodeid1, name1, !!wnode1);
1366 debug_path(f, " PATH2", nodeid2, name2, !!wnode2);
1367 err = wait_path(f, &qe);
1368 debug_path(f, "DEQUEUE PATH1", nodeid1, name1, !!wnode1);
1369 debug_path(f, " PATH2", nodeid2, name2, !!wnode2);
1370 }
1371
1372 #if defined(CHECK_DIR_LOOP)
1373 out_unlock:
1374 #endif
1375 pthread_mutex_unlock(&f->lock);
1376
1377 return err;
1378 }
1379
free_path_wrlock(struct fuse * f,fuse_ino_t nodeid,struct node * wnode,char * path)1380 static void free_path_wrlock(struct fuse *f, fuse_ino_t nodeid,
1381 struct node *wnode, char *path)
1382 {
1383 pthread_mutex_lock(&f->lock);
1384 unlock_path(f, nodeid, wnode, NULL);
1385 if (f->lockq)
1386 wake_up_queued(f);
1387 pthread_mutex_unlock(&f->lock);
1388 free(path);
1389 }
1390
free_path(struct fuse * f,fuse_ino_t nodeid,char * path)1391 static void free_path(struct fuse *f, fuse_ino_t nodeid, char *path)
1392 {
1393 if (path)
1394 free_path_wrlock(f, nodeid, NULL, path);
1395 }
1396
free_path2(struct fuse * f,fuse_ino_t nodeid1,fuse_ino_t nodeid2,struct node * wnode1,struct node * wnode2,char * path1,char * path2)1397 static void free_path2(struct fuse *f, fuse_ino_t nodeid1, fuse_ino_t nodeid2,
1398 struct node *wnode1, struct node *wnode2,
1399 char *path1, char *path2)
1400 {
1401 pthread_mutex_lock(&f->lock);
1402 unlock_path(f, nodeid1, wnode1, NULL);
1403 unlock_path(f, nodeid2, wnode2, NULL);
1404 wake_up_queued(f);
1405 pthread_mutex_unlock(&f->lock);
1406 free(path1);
1407 free(path2);
1408 }
1409
forget_node(struct fuse * f,fuse_ino_t nodeid,uint64_t nlookup)1410 static void forget_node(struct fuse *f, fuse_ino_t nodeid, uint64_t nlookup)
1411 {
1412 struct node *node;
1413 if (nodeid == FUSE_ROOT_ID)
1414 return;
1415 pthread_mutex_lock(&f->lock);
1416 node = get_node(f, nodeid);
1417
1418 /*
1419 * Node may still be locked due to interrupt idiocy in open,
1420 * create and opendir
1421 */
1422 while (node->nlookup == nlookup && node->treelock) {
1423 struct lock_queue_element qe = {
1424 .nodeid1 = nodeid,
1425 };
1426
1427 debug_path(f, "QUEUE PATH (forget)", nodeid, NULL, false);
1428 queue_path(f, &qe);
1429
1430 do {
1431 pthread_cond_wait(&qe.cond, &f->lock);
1432 } while (node->nlookup == nlookup && node->treelock);
1433
1434 dequeue_path(f, &qe);
1435 debug_path(f, "DEQUEUE_PATH (forget)", nodeid, NULL, false);
1436 }
1437
1438 assert(node->nlookup >= nlookup);
1439 node->nlookup -= nlookup;
1440 if (!node->nlookup) {
1441 unref_node(f, node);
1442 } else if (lru_enabled(f) && node->nlookup == 1) {
1443 set_forget_time(f, node);
1444 }
1445 pthread_mutex_unlock(&f->lock);
1446 }
1447
unlink_node(struct fuse * f,struct node * node)1448 static void unlink_node(struct fuse *f, struct node *node)
1449 {
1450 if (f->conf.remember) {
1451 assert(node->nlookup > 1);
1452 node->nlookup--;
1453 }
1454 unhash_name(f, node);
1455 }
1456
remove_node(struct fuse * f,fuse_ino_t dir,const char * name)1457 static void remove_node(struct fuse *f, fuse_ino_t dir, const char *name)
1458 {
1459 struct node *node;
1460
1461 pthread_mutex_lock(&f->lock);
1462 node = lookup_node(f, dir, name);
1463 if (node != NULL)
1464 unlink_node(f, node);
1465 pthread_mutex_unlock(&f->lock);
1466 }
1467
rename_node(struct fuse * f,fuse_ino_t olddir,const char * oldname,fuse_ino_t newdir,const char * newname,int hide)1468 static int rename_node(struct fuse *f, fuse_ino_t olddir, const char *oldname,
1469 fuse_ino_t newdir, const char *newname, int hide)
1470 {
1471 struct node *node;
1472 struct node *newnode;
1473 int err = 0;
1474
1475 pthread_mutex_lock(&f->lock);
1476 node = lookup_node(f, olddir, oldname);
1477 newnode = lookup_node(f, newdir, newname);
1478 if (node == NULL)
1479 goto out;
1480
1481 if (newnode != NULL) {
1482 if (hide) {
1483 fuse_log(FUSE_LOG_ERR, "fuse: hidden file got created during hiding\n");
1484 err = -EBUSY;
1485 goto out;
1486 }
1487 unlink_node(f, newnode);
1488 }
1489
1490 unhash_name(f, node);
1491 if (hash_name(f, node, newdir, newname) == -1) {
1492 err = -ENOMEM;
1493 goto out;
1494 }
1495
1496 if (hide)
1497 node->is_hidden = 1;
1498
1499 out:
1500 pthread_mutex_unlock(&f->lock);
1501 return err;
1502 }
1503
exchange_node(struct fuse * f,fuse_ino_t olddir,const char * oldname,fuse_ino_t newdir,const char * newname)1504 static int exchange_node(struct fuse *f, fuse_ino_t olddir, const char *oldname,
1505 fuse_ino_t newdir, const char *newname)
1506 {
1507 struct node *oldnode;
1508 struct node *newnode;
1509 int err;
1510
1511 pthread_mutex_lock(&f->lock);
1512 oldnode = lookup_node(f, olddir, oldname);
1513 newnode = lookup_node(f, newdir, newname);
1514
1515 if (oldnode)
1516 unhash_name(f, oldnode);
1517 if (newnode)
1518 unhash_name(f, newnode);
1519
1520 err = -ENOMEM;
1521 if (oldnode) {
1522 if (hash_name(f, oldnode, newdir, newname) == -1)
1523 goto out;
1524 }
1525 if (newnode) {
1526 if (hash_name(f, newnode, olddir, oldname) == -1)
1527 goto out;
1528 }
1529 err = 0;
1530 out:
1531 pthread_mutex_unlock(&f->lock);
1532 return err;
1533 }
1534
set_stat(struct fuse * f,fuse_ino_t nodeid,struct stat * stbuf)1535 static void set_stat(struct fuse *f, fuse_ino_t nodeid, struct stat *stbuf)
1536 {
1537 if (!f->conf.use_ino)
1538 stbuf->st_ino = nodeid;
1539 if (f->conf.set_mode)
1540 stbuf->st_mode = (stbuf->st_mode & S_IFMT) |
1541 (0777 & ~f->conf.umask);
1542 if (f->conf.set_uid)
1543 stbuf->st_uid = f->conf.uid;
1544 if (f->conf.set_gid)
1545 stbuf->st_gid = f->conf.gid;
1546 }
1547
req_fuse(fuse_req_t req)1548 static struct fuse *req_fuse(fuse_req_t req)
1549 {
1550 return (struct fuse *) fuse_req_userdata(req);
1551 }
1552
fuse_intr_sighandler(int sig)1553 static void fuse_intr_sighandler(int sig)
1554 {
1555 (void) sig;
1556 /* Nothing to do */
1557 }
1558
1559 struct fuse_intr_data {
1560 pthread_t id;
1561 pthread_cond_t cond;
1562 int finished;
1563 };
1564
fuse_interrupt(fuse_req_t req,void * d_)1565 static void fuse_interrupt(fuse_req_t req, void *d_)
1566 {
1567 struct fuse_intr_data *d = d_;
1568 struct fuse *f = req_fuse(req);
1569
1570 if (d->id == pthread_self())
1571 return;
1572
1573 pthread_mutex_lock(&f->lock);
1574 while (!d->finished) {
1575 struct timeval now;
1576 struct timespec timeout;
1577
1578 pthread_kill(d->id, f->conf.intr_signal);
1579 gettimeofday(&now, NULL);
1580 timeout.tv_sec = now.tv_sec + 1;
1581 timeout.tv_nsec = now.tv_usec * 1000;
1582 pthread_cond_timedwait(&d->cond, &f->lock, &timeout);
1583 }
1584 pthread_mutex_unlock(&f->lock);
1585 }
1586
fuse_do_finish_interrupt(struct fuse * f,fuse_req_t req,struct fuse_intr_data * d)1587 static void fuse_do_finish_interrupt(struct fuse *f, fuse_req_t req,
1588 struct fuse_intr_data *d)
1589 {
1590 pthread_mutex_lock(&f->lock);
1591 d->finished = 1;
1592 pthread_cond_broadcast(&d->cond);
1593 pthread_mutex_unlock(&f->lock);
1594 fuse_req_interrupt_func(req, NULL, NULL);
1595 pthread_cond_destroy(&d->cond);
1596 }
1597
fuse_do_prepare_interrupt(fuse_req_t req,struct fuse_intr_data * d)1598 static void fuse_do_prepare_interrupt(fuse_req_t req, struct fuse_intr_data *d)
1599 {
1600 d->id = pthread_self();
1601 pthread_cond_init(&d->cond, NULL);
1602 d->finished = 0;
1603 fuse_req_interrupt_func(req, fuse_interrupt, d);
1604 }
1605
fuse_finish_interrupt(struct fuse * f,fuse_req_t req,struct fuse_intr_data * d)1606 static inline void fuse_finish_interrupt(struct fuse *f, fuse_req_t req,
1607 struct fuse_intr_data *d)
1608 {
1609 if (f->conf.intr)
1610 fuse_do_finish_interrupt(f, req, d);
1611 }
1612
fuse_prepare_interrupt(struct fuse * f,fuse_req_t req,struct fuse_intr_data * d)1613 static inline void fuse_prepare_interrupt(struct fuse *f, fuse_req_t req,
1614 struct fuse_intr_data *d)
1615 {
1616 if (f->conf.intr)
1617 fuse_do_prepare_interrupt(req, d);
1618 }
1619
file_info_string(struct fuse_file_info * fi,char * buf,size_t len)1620 static const char* file_info_string(struct fuse_file_info *fi,
1621 char* buf, size_t len)
1622 {
1623 if(fi == NULL)
1624 return "NULL";
1625 snprintf(buf, len, "%llu", (unsigned long long) fi->fh);
1626 return buf;
1627 }
1628
fuse_fs_getattr(struct fuse_fs * fs,const char * path,struct stat * buf,struct fuse_file_info * fi)1629 int fuse_fs_getattr(struct fuse_fs *fs, const char *path, struct stat *buf,
1630 struct fuse_file_info *fi)
1631 {
1632 fuse_get_context()->private_data = fs->user_data;
1633 if (fs->op.getattr) {
1634 if (fs->debug) {
1635 char buf[10];
1636 fuse_log(FUSE_LOG_DEBUG, "getattr[%s] %s\n",
1637 file_info_string(fi, buf, sizeof(buf)),
1638 path);
1639 }
1640 return fs->op.getattr(path, buf, fi);
1641 } else {
1642 return -ENOSYS;
1643 }
1644 }
1645
fuse_fs_rename(struct fuse_fs * fs,const char * oldpath,const char * newpath,unsigned int flags)1646 int fuse_fs_rename(struct fuse_fs *fs, const char *oldpath,
1647 const char *newpath, unsigned int flags)
1648 {
1649 fuse_get_context()->private_data = fs->user_data;
1650 if (fs->op.rename) {
1651 if (fs->debug)
1652 fuse_log(FUSE_LOG_DEBUG, "rename %s %s 0x%x\n", oldpath, newpath,
1653 flags);
1654
1655 return fs->op.rename(oldpath, newpath, flags);
1656 } else {
1657 return -ENOSYS;
1658 }
1659 }
1660
fuse_fs_unlink(struct fuse_fs * fs,const char * path)1661 int fuse_fs_unlink(struct fuse_fs *fs, const char *path)
1662 {
1663 fuse_get_context()->private_data = fs->user_data;
1664 if (fs->op.unlink) {
1665 if (fs->debug)
1666 fuse_log(FUSE_LOG_DEBUG, "unlink %s\n", path);
1667
1668 return fs->op.unlink(path);
1669 } else {
1670 return -ENOSYS;
1671 }
1672 }
1673
fuse_fs_rmdir(struct fuse_fs * fs,const char * path)1674 int fuse_fs_rmdir(struct fuse_fs *fs, const char *path)
1675 {
1676 fuse_get_context()->private_data = fs->user_data;
1677 if (fs->op.rmdir) {
1678 if (fs->debug)
1679 fuse_log(FUSE_LOG_DEBUG, "rmdir %s\n", path);
1680
1681 return fs->op.rmdir(path);
1682 } else {
1683 return -ENOSYS;
1684 }
1685 }
1686
fuse_fs_symlink(struct fuse_fs * fs,const char * linkname,const char * path)1687 int fuse_fs_symlink(struct fuse_fs *fs, const char *linkname, const char *path)
1688 {
1689 fuse_get_context()->private_data = fs->user_data;
1690 if (fs->op.symlink) {
1691 if (fs->debug)
1692 fuse_log(FUSE_LOG_DEBUG, "symlink %s %s\n", linkname, path);
1693
1694 return fs->op.symlink(linkname, path);
1695 } else {
1696 return -ENOSYS;
1697 }
1698 }
1699
fuse_fs_link(struct fuse_fs * fs,const char * oldpath,const char * newpath)1700 int fuse_fs_link(struct fuse_fs *fs, const char *oldpath, const char *newpath)
1701 {
1702 fuse_get_context()->private_data = fs->user_data;
1703 if (fs->op.link) {
1704 if (fs->debug)
1705 fuse_log(FUSE_LOG_DEBUG, "link %s %s\n", oldpath, newpath);
1706
1707 return fs->op.link(oldpath, newpath);
1708 } else {
1709 return -ENOSYS;
1710 }
1711 }
1712
fuse_fs_release(struct fuse_fs * fs,const char * path,struct fuse_file_info * fi)1713 int fuse_fs_release(struct fuse_fs *fs, const char *path,
1714 struct fuse_file_info *fi)
1715 {
1716 fuse_get_context()->private_data = fs->user_data;
1717 if (fs->op.release) {
1718 if (fs->debug)
1719 fuse_log(FUSE_LOG_DEBUG, "release%s[%llu] flags: 0x%x\n",
1720 fi->flush ? "+flush" : "",
1721 (unsigned long long) fi->fh, fi->flags);
1722
1723 return fs->op.release(path, fi);
1724 } else {
1725 return 0;
1726 }
1727 }
1728
fuse_fs_opendir(struct fuse_fs * fs,const char * path,struct fuse_file_info * fi)1729 int fuse_fs_opendir(struct fuse_fs *fs, const char *path,
1730 struct fuse_file_info *fi)
1731 {
1732 fuse_get_context()->private_data = fs->user_data;
1733 if (fs->op.opendir) {
1734 int err;
1735
1736 if (fs->debug)
1737 fuse_log(FUSE_LOG_DEBUG, "opendir flags: 0x%x %s\n", fi->flags,
1738 path);
1739
1740 err = fs->op.opendir(path, fi);
1741
1742 if (fs->debug && !err)
1743 fuse_log(FUSE_LOG_DEBUG, " opendir[%llu] flags: 0x%x %s\n",
1744 (unsigned long long) fi->fh, fi->flags, path);
1745
1746 return err;
1747 } else {
1748 return 0;
1749 }
1750 }
1751
fuse_fs_open(struct fuse_fs * fs,const char * path,struct fuse_file_info * fi)1752 int fuse_fs_open(struct fuse_fs *fs, const char *path,
1753 struct fuse_file_info *fi)
1754 {
1755 fuse_get_context()->private_data = fs->user_data;
1756 if (fs->op.open) {
1757 int err;
1758
1759 if (fs->debug)
1760 fuse_log(FUSE_LOG_DEBUG, "open flags: 0x%x %s\n", fi->flags,
1761 path);
1762
1763 err = fs->op.open(path, fi);
1764
1765 if (fs->debug && !err)
1766 fuse_log(FUSE_LOG_DEBUG, " open[%llu] flags: 0x%x %s\n",
1767 (unsigned long long) fi->fh, fi->flags, path);
1768
1769 return err;
1770 } else {
1771 return 0;
1772 }
1773 }
1774
fuse_free_buf(struct fuse_bufvec * buf)1775 static void fuse_free_buf(struct fuse_bufvec *buf)
1776 {
1777 if (buf != NULL) {
1778 size_t i;
1779
1780 for (i = 0; i < buf->count; i++)
1781 if (!(buf->buf[i].flags & FUSE_BUF_IS_FD))
1782 free(buf->buf[i].mem);
1783 free(buf);
1784 }
1785 }
1786
fuse_fs_read_buf(struct fuse_fs * fs,const char * path,struct fuse_bufvec ** bufp,size_t size,off_t off,struct fuse_file_info * fi)1787 int fuse_fs_read_buf(struct fuse_fs *fs, const char *path,
1788 struct fuse_bufvec **bufp, size_t size, off_t off,
1789 struct fuse_file_info *fi)
1790 {
1791 fuse_get_context()->private_data = fs->user_data;
1792 if (fs->op.read || fs->op.read_buf) {
1793 int res;
1794
1795 if (fs->debug)
1796 fuse_log(FUSE_LOG_DEBUG,
1797 "read[%llu] %zu bytes from %llu flags: 0x%x\n",
1798 (unsigned long long) fi->fh,
1799 size, (unsigned long long) off, fi->flags);
1800
1801 if (fs->op.read_buf) {
1802 res = fs->op.read_buf(path, bufp, size, off, fi);
1803 } else {
1804 struct fuse_bufvec *buf;
1805 void *mem;
1806
1807 buf = malloc(sizeof(struct fuse_bufvec));
1808 if (buf == NULL)
1809 return -ENOMEM;
1810
1811 mem = malloc(size);
1812 if (mem == NULL) {
1813 free(buf);
1814 return -ENOMEM;
1815 }
1816 *buf = FUSE_BUFVEC_INIT(size);
1817 buf->buf[0].mem = mem;
1818 *bufp = buf;
1819
1820 res = fs->op.read(path, mem, size, off, fi);
1821 if (res >= 0)
1822 buf->buf[0].size = res;
1823 }
1824
1825 if (fs->debug && res >= 0)
1826 fuse_log(FUSE_LOG_DEBUG, " read[%llu] %zu bytes from %llu\n",
1827 (unsigned long long) fi->fh,
1828 fuse_buf_size(*bufp),
1829 (unsigned long long) off);
1830 if (res >= 0 && fuse_buf_size(*bufp) > size)
1831 fuse_log(FUSE_LOG_ERR, "fuse: read too many bytes\n");
1832
1833 if (res < 0)
1834 return res;
1835
1836 return 0;
1837 } else {
1838 return -ENOSYS;
1839 }
1840 }
1841
fuse_fs_read(struct fuse_fs * fs,const char * path,char * mem,size_t size,off_t off,struct fuse_file_info * fi)1842 int fuse_fs_read(struct fuse_fs *fs, const char *path, char *mem, size_t size,
1843 off_t off, struct fuse_file_info *fi)
1844 {
1845 fuse_get_context()->private_data = fs->user_data;
1846 if (fs->op.read || fs->op.read_buf) {
1847 int res;
1848
1849 if (fs->debug)
1850 fuse_log(FUSE_LOG_DEBUG,
1851 "read[%llu] %zu bytes from %llu flags: 0x%x\n",
1852 (unsigned long long) fi->fh,
1853 size, (unsigned long long) off, fi->flags);
1854
1855 if (fs->op.read_buf) {
1856 struct fuse_bufvec *buf = NULL;
1857
1858 res = fs->op.read_buf(path, &buf, size, off, fi);
1859 if (res == 0) {
1860 struct fuse_bufvec dst = FUSE_BUFVEC_INIT(size);
1861
1862 dst.buf[0].mem = mem;
1863 res = fuse_buf_copy(&dst, buf, 0);
1864 }
1865 fuse_free_buf(buf);
1866 } else {
1867 res = fs->op.read(path, mem, size, off, fi);
1868 }
1869
1870 if (fs->debug && res >= 0)
1871 fuse_log(FUSE_LOG_DEBUG, " read[%llu] %u bytes from %llu\n",
1872 (unsigned long long) fi->fh,
1873 res,
1874 (unsigned long long) off);
1875 if (res >= 0 && res > (int) size)
1876 fuse_log(FUSE_LOG_ERR, "fuse: read too many bytes\n");
1877
1878 return res;
1879 } else {
1880 return -ENOSYS;
1881 }
1882 }
1883
fuse_fs_write_buf(struct fuse_fs * fs,const char * path,struct fuse_bufvec * buf,off_t off,struct fuse_file_info * fi)1884 int fuse_fs_write_buf(struct fuse_fs *fs, const char *path,
1885 struct fuse_bufvec *buf, off_t off,
1886 struct fuse_file_info *fi)
1887 {
1888 fuse_get_context()->private_data = fs->user_data;
1889 if (fs->op.write_buf || fs->op.write) {
1890 int res;
1891 size_t size = fuse_buf_size(buf);
1892
1893 assert(buf->idx == 0 && buf->off == 0);
1894 if (fs->debug)
1895 fuse_log(FUSE_LOG_DEBUG,
1896 "write%s[%llu] %zu bytes to %llu flags: 0x%x\n",
1897 fi->writepage ? "page" : "",
1898 (unsigned long long) fi->fh,
1899 size,
1900 (unsigned long long) off,
1901 fi->flags);
1902
1903 if (fs->op.write_buf) {
1904 res = fs->op.write_buf(path, buf, off, fi);
1905 } else {
1906 void *mem = NULL;
1907 struct fuse_buf *flatbuf;
1908 struct fuse_bufvec tmp = FUSE_BUFVEC_INIT(size);
1909
1910 if (buf->count == 1 &&
1911 !(buf->buf[0].flags & FUSE_BUF_IS_FD)) {
1912 flatbuf = &buf->buf[0];
1913 } else {
1914 res = -ENOMEM;
1915 mem = malloc(size);
1916 if (mem == NULL)
1917 goto out;
1918
1919 tmp.buf[0].mem = mem;
1920 res = fuse_buf_copy(&tmp, buf, 0);
1921 if (res <= 0)
1922 goto out_free;
1923
1924 tmp.buf[0].size = res;
1925 flatbuf = &tmp.buf[0];
1926 }
1927
1928 res = fs->op.write(path, flatbuf->mem, flatbuf->size,
1929 off, fi);
1930 out_free:
1931 free(mem);
1932 }
1933 out:
1934 if (fs->debug && res >= 0)
1935 fuse_log(FUSE_LOG_DEBUG, " write%s[%llu] %u bytes to %llu\n",
1936 fi->writepage ? "page" : "",
1937 (unsigned long long) fi->fh, res,
1938 (unsigned long long) off);
1939 if (res > (int) size)
1940 fuse_log(FUSE_LOG_ERR, "fuse: wrote too many bytes\n");
1941
1942 return res;
1943 } else {
1944 return -ENOSYS;
1945 }
1946 }
1947
fuse_fs_write(struct fuse_fs * fs,const char * path,const char * mem,size_t size,off_t off,struct fuse_file_info * fi)1948 int fuse_fs_write(struct fuse_fs *fs, const char *path, const char *mem,
1949 size_t size, off_t off, struct fuse_file_info *fi)
1950 {
1951 struct fuse_bufvec bufv = FUSE_BUFVEC_INIT(size);
1952
1953 bufv.buf[0].mem = (void *) mem;
1954
1955 return fuse_fs_write_buf(fs, path, &bufv, off, fi);
1956 }
1957
fuse_fs_fsync(struct fuse_fs * fs,const char * path,int datasync,struct fuse_file_info * fi)1958 int fuse_fs_fsync(struct fuse_fs *fs, const char *path, int datasync,
1959 struct fuse_file_info *fi)
1960 {
1961 fuse_get_context()->private_data = fs->user_data;
1962 if (fs->op.fsync) {
1963 if (fs->debug)
1964 fuse_log(FUSE_LOG_DEBUG, "fsync[%llu] datasync: %i\n",
1965 (unsigned long long) fi->fh, datasync);
1966
1967 return fs->op.fsync(path, datasync, fi);
1968 } else {
1969 return -ENOSYS;
1970 }
1971 }
1972
fuse_fs_fsyncdir(struct fuse_fs * fs,const char * path,int datasync,struct fuse_file_info * fi)1973 int fuse_fs_fsyncdir(struct fuse_fs *fs, const char *path, int datasync,
1974 struct fuse_file_info *fi)
1975 {
1976 fuse_get_context()->private_data = fs->user_data;
1977 if (fs->op.fsyncdir) {
1978 if (fs->debug)
1979 fuse_log(FUSE_LOG_DEBUG, "fsyncdir[%llu] datasync: %i\n",
1980 (unsigned long long) fi->fh, datasync);
1981
1982 return fs->op.fsyncdir(path, datasync, fi);
1983 } else {
1984 return -ENOSYS;
1985 }
1986 }
1987
fuse_fs_flush(struct fuse_fs * fs,const char * path,struct fuse_file_info * fi)1988 int fuse_fs_flush(struct fuse_fs *fs, const char *path,
1989 struct fuse_file_info *fi)
1990 {
1991 fuse_get_context()->private_data = fs->user_data;
1992 if (fs->op.flush) {
1993 if (fs->debug)
1994 fuse_log(FUSE_LOG_DEBUG, "flush[%llu]\n",
1995 (unsigned long long) fi->fh);
1996
1997 return fs->op.flush(path, fi);
1998 } else {
1999 return -ENOSYS;
2000 }
2001 }
2002
fuse_fs_statfs(struct fuse_fs * fs,const char * path,struct statvfs * buf)2003 int fuse_fs_statfs(struct fuse_fs *fs, const char *path, struct statvfs *buf)
2004 {
2005 fuse_get_context()->private_data = fs->user_data;
2006 if (fs->op.statfs) {
2007 if (fs->debug)
2008 fuse_log(FUSE_LOG_DEBUG, "statfs %s\n", path);
2009
2010 return fs->op.statfs(path, buf);
2011 } else {
2012 buf->f_namemax = 255;
2013 buf->f_bsize = 512;
2014 return 0;
2015 }
2016 }
2017
fuse_fs_releasedir(struct fuse_fs * fs,const char * path,struct fuse_file_info * fi)2018 int fuse_fs_releasedir(struct fuse_fs *fs, const char *path,
2019 struct fuse_file_info *fi)
2020 {
2021 fuse_get_context()->private_data = fs->user_data;
2022 if (fs->op.releasedir) {
2023 if (fs->debug)
2024 fuse_log(FUSE_LOG_DEBUG, "releasedir[%llu] flags: 0x%x\n",
2025 (unsigned long long) fi->fh, fi->flags);
2026
2027 return fs->op.releasedir(path, fi);
2028 } else {
2029 return 0;
2030 }
2031 }
2032
fuse_fs_readdir(struct fuse_fs * fs,const char * path,void * buf,fuse_fill_dir_t filler,off_t off,struct fuse_file_info * fi,enum fuse_readdir_flags flags)2033 int fuse_fs_readdir(struct fuse_fs *fs, const char *path, void *buf,
2034 fuse_fill_dir_t filler, off_t off,
2035 struct fuse_file_info *fi,
2036 enum fuse_readdir_flags flags)
2037 {
2038 fuse_get_context()->private_data = fs->user_data;
2039 if (fs->op.readdir) {
2040 if (fs->debug) {
2041 fuse_log(FUSE_LOG_DEBUG, "readdir%s[%llu] from %llu\n",
2042 (flags & FUSE_READDIR_PLUS) ? "plus" : "",
2043 (unsigned long long) fi->fh,
2044 (unsigned long long) off);
2045 }
2046
2047 return fs->op.readdir(path, buf, filler, off, fi, flags);
2048 } else {
2049 return -ENOSYS;
2050 }
2051 }
2052
fuse_fs_create(struct fuse_fs * fs,const char * path,mode_t mode,struct fuse_file_info * fi)2053 int fuse_fs_create(struct fuse_fs *fs, const char *path, mode_t mode,
2054 struct fuse_file_info *fi)
2055 {
2056 fuse_get_context()->private_data = fs->user_data;
2057 if (fs->op.create) {
2058 int err;
2059
2060 if (fs->debug)
2061 fuse_log(FUSE_LOG_DEBUG,
2062 "create flags: 0x%x %s 0%o umask=0%03o\n",
2063 fi->flags, path, mode,
2064 fuse_get_context()->umask);
2065
2066 err = fs->op.create(path, mode, fi);
2067
2068 if (fs->debug && !err)
2069 fuse_log(FUSE_LOG_DEBUG, " create[%llu] flags: 0x%x %s\n",
2070 (unsigned long long) fi->fh, fi->flags, path);
2071
2072 return err;
2073 } else {
2074 return -ENOSYS;
2075 }
2076 }
2077
fuse_fs_lock(struct fuse_fs * fs,const char * path,struct fuse_file_info * fi,int cmd,struct flock * lock)2078 int fuse_fs_lock(struct fuse_fs *fs, const char *path,
2079 struct fuse_file_info *fi, int cmd, struct flock *lock)
2080 {
2081 fuse_get_context()->private_data = fs->user_data;
2082 if (fs->op.lock) {
2083 if (fs->debug)
2084 fuse_log(FUSE_LOG_DEBUG, "lock[%llu] %s %s start: %llu len: %llu pid: %llu\n",
2085 (unsigned long long) fi->fh,
2086 (cmd == F_GETLK ? "F_GETLK" :
2087 (cmd == F_SETLK ? "F_SETLK" :
2088 (cmd == F_SETLKW ? "F_SETLKW" : "???"))),
2089 (lock->l_type == F_RDLCK ? "F_RDLCK" :
2090 (lock->l_type == F_WRLCK ? "F_WRLCK" :
2091 (lock->l_type == F_UNLCK ? "F_UNLCK" :
2092 "???"))),
2093 (unsigned long long) lock->l_start,
2094 (unsigned long long) lock->l_len,
2095 (unsigned long long) lock->l_pid);
2096
2097 return fs->op.lock(path, fi, cmd, lock);
2098 } else {
2099 return -ENOSYS;
2100 }
2101 }
2102
fuse_fs_flock(struct fuse_fs * fs,const char * path,struct fuse_file_info * fi,int op)2103 int fuse_fs_flock(struct fuse_fs *fs, const char *path,
2104 struct fuse_file_info *fi, int op)
2105 {
2106 fuse_get_context()->private_data = fs->user_data;
2107 if (fs->op.flock) {
2108 if (fs->debug) {
2109 int xop = op & ~LOCK_NB;
2110
2111 fuse_log(FUSE_LOG_DEBUG, "lock[%llu] %s%s\n",
2112 (unsigned long long) fi->fh,
2113 xop == LOCK_SH ? "LOCK_SH" :
2114 (xop == LOCK_EX ? "LOCK_EX" :
2115 (xop == LOCK_UN ? "LOCK_UN" : "???")),
2116 (op & LOCK_NB) ? "|LOCK_NB" : "");
2117 }
2118 return fs->op.flock(path, fi, op);
2119 } else {
2120 return -ENOSYS;
2121 }
2122 }
2123
fuse_fs_chown(struct fuse_fs * fs,const char * path,uid_t uid,gid_t gid,struct fuse_file_info * fi)2124 int fuse_fs_chown(struct fuse_fs *fs, const char *path, uid_t uid,
2125 gid_t gid, struct fuse_file_info *fi)
2126 {
2127 fuse_get_context()->private_data = fs->user_data;
2128 if (fs->op.chown) {
2129 if (fs->debug) {
2130 char buf[10];
2131 fuse_log(FUSE_LOG_DEBUG, "chown[%s] %s %lu %lu\n",
2132 file_info_string(fi, buf, sizeof(buf)),
2133 path, (unsigned long) uid, (unsigned long) gid);
2134 }
2135 return fs->op.chown(path, uid, gid, fi);
2136 } else {
2137 return -ENOSYS;
2138 }
2139 }
2140
fuse_fs_truncate(struct fuse_fs * fs,const char * path,off_t size,struct fuse_file_info * fi)2141 int fuse_fs_truncate(struct fuse_fs *fs, const char *path, off_t size,
2142 struct fuse_file_info *fi)
2143 {
2144 fuse_get_context()->private_data = fs->user_data;
2145 if (fs->op.truncate) {
2146 if (fs->debug) {
2147 char buf[10];
2148 fuse_log(FUSE_LOG_DEBUG, "truncate[%s] %llu\n",
2149 file_info_string(fi, buf, sizeof(buf)),
2150 (unsigned long long) size);
2151 }
2152 return fs->op.truncate(path, size, fi);
2153 } else {
2154 return -ENOSYS;
2155 }
2156 }
2157
fuse_fs_utimens(struct fuse_fs * fs,const char * path,const struct timespec tv[2],struct fuse_file_info * fi)2158 int fuse_fs_utimens(struct fuse_fs *fs, const char *path,
2159 const struct timespec tv[2], struct fuse_file_info *fi)
2160 {
2161 fuse_get_context()->private_data = fs->user_data;
2162 if (fs->op.utimens) {
2163 if (fs->debug) {
2164 char buf[10];
2165 fuse_log(FUSE_LOG_DEBUG, "utimens[%s] %s %li.%09lu %li.%09lu\n",
2166 file_info_string(fi, buf, sizeof(buf)),
2167 path, tv[0].tv_sec, tv[0].tv_nsec,
2168 tv[1].tv_sec, tv[1].tv_nsec);
2169 }
2170 return fs->op.utimens(path, tv, fi);
2171 } else {
2172 return -ENOSYS;
2173 }
2174 }
2175
fuse_fs_access(struct fuse_fs * fs,const char * path,int mask)2176 int fuse_fs_access(struct fuse_fs *fs, const char *path, int mask)
2177 {
2178 fuse_get_context()->private_data = fs->user_data;
2179 if (fs->op.access) {
2180 if (fs->debug)
2181 fuse_log(FUSE_LOG_DEBUG, "access %s 0%o\n", path, mask);
2182
2183 return fs->op.access(path, mask);
2184 } else {
2185 return -ENOSYS;
2186 }
2187 }
2188
fuse_fs_readlink(struct fuse_fs * fs,const char * path,char * buf,size_t len)2189 int fuse_fs_readlink(struct fuse_fs *fs, const char *path, char *buf,
2190 size_t len)
2191 {
2192 fuse_get_context()->private_data = fs->user_data;
2193 if (fs->op.readlink) {
2194 if (fs->debug)
2195 fuse_log(FUSE_LOG_DEBUG, "readlink %s %lu\n", path,
2196 (unsigned long) len);
2197
2198 return fs->op.readlink(path, buf, len);
2199 } else {
2200 return -ENOSYS;
2201 }
2202 }
2203
fuse_fs_mknod(struct fuse_fs * fs,const char * path,mode_t mode,dev_t rdev)2204 int fuse_fs_mknod(struct fuse_fs *fs, const char *path, mode_t mode,
2205 dev_t rdev)
2206 {
2207 fuse_get_context()->private_data = fs->user_data;
2208 if (fs->op.mknod) {
2209 if (fs->debug)
2210 fuse_log(FUSE_LOG_DEBUG, "mknod %s 0%o 0x%llx umask=0%03o\n",
2211 path, mode, (unsigned long long) rdev,
2212 fuse_get_context()->umask);
2213
2214 return fs->op.mknod(path, mode, rdev);
2215 } else {
2216 return -ENOSYS;
2217 }
2218 }
2219
fuse_fs_mkdir(struct fuse_fs * fs,const char * path,mode_t mode)2220 int fuse_fs_mkdir(struct fuse_fs *fs, const char *path, mode_t mode)
2221 {
2222 fuse_get_context()->private_data = fs->user_data;
2223 if (fs->op.mkdir) {
2224 if (fs->debug)
2225 fuse_log(FUSE_LOG_DEBUG, "mkdir %s 0%o umask=0%03o\n",
2226 path, mode, fuse_get_context()->umask);
2227
2228 return fs->op.mkdir(path, mode);
2229 } else {
2230 return -ENOSYS;
2231 }
2232 }
2233
fuse_fs_setxattr(struct fuse_fs * fs,const char * path,const char * name,const char * value,size_t size,int flags)2234 int fuse_fs_setxattr(struct fuse_fs *fs, const char *path, const char *name,
2235 const char *value, size_t size, int flags)
2236 {
2237 fuse_get_context()->private_data = fs->user_data;
2238 if (fs->op.setxattr) {
2239 if (fs->debug)
2240 fuse_log(FUSE_LOG_DEBUG, "setxattr %s %s %lu 0x%x\n",
2241 path, name, (unsigned long) size, flags);
2242
2243 return fs->op.setxattr(path, name, value, size, flags);
2244 } else {
2245 return -ENOSYS;
2246 }
2247 }
2248
fuse_fs_getxattr(struct fuse_fs * fs,const char * path,const char * name,char * value,size_t size)2249 int fuse_fs_getxattr(struct fuse_fs *fs, const char *path, const char *name,
2250 char *value, size_t size)
2251 {
2252 fuse_get_context()->private_data = fs->user_data;
2253 if (fs->op.getxattr) {
2254 if (fs->debug)
2255 fuse_log(FUSE_LOG_DEBUG, "getxattr %s %s %lu\n",
2256 path, name, (unsigned long) size);
2257
2258 return fs->op.getxattr(path, name, value, size);
2259 } else {
2260 return -ENOSYS;
2261 }
2262 }
2263
fuse_fs_listxattr(struct fuse_fs * fs,const char * path,char * list,size_t size)2264 int fuse_fs_listxattr(struct fuse_fs *fs, const char *path, char *list,
2265 size_t size)
2266 {
2267 fuse_get_context()->private_data = fs->user_data;
2268 if (fs->op.listxattr) {
2269 if (fs->debug)
2270 fuse_log(FUSE_LOG_DEBUG, "listxattr %s %lu\n",
2271 path, (unsigned long) size);
2272
2273 return fs->op.listxattr(path, list, size);
2274 } else {
2275 return -ENOSYS;
2276 }
2277 }
2278
fuse_fs_bmap(struct fuse_fs * fs,const char * path,size_t blocksize,uint64_t * idx)2279 int fuse_fs_bmap(struct fuse_fs *fs, const char *path, size_t blocksize,
2280 uint64_t *idx)
2281 {
2282 fuse_get_context()->private_data = fs->user_data;
2283 if (fs->op.bmap) {
2284 if (fs->debug)
2285 fuse_log(FUSE_LOG_DEBUG, "bmap %s blocksize: %lu index: %llu\n",
2286 path, (unsigned long) blocksize,
2287 (unsigned long long) *idx);
2288
2289 return fs->op.bmap(path, blocksize, idx);
2290 } else {
2291 return -ENOSYS;
2292 }
2293 }
2294
fuse_fs_removexattr(struct fuse_fs * fs,const char * path,const char * name)2295 int fuse_fs_removexattr(struct fuse_fs *fs, const char *path, const char *name)
2296 {
2297 fuse_get_context()->private_data = fs->user_data;
2298 if (fs->op.removexattr) {
2299 if (fs->debug)
2300 fuse_log(FUSE_LOG_DEBUG, "removexattr %s %s\n", path, name);
2301
2302 return fs->op.removexattr(path, name);
2303 } else {
2304 return -ENOSYS;
2305 }
2306 }
2307
fuse_fs_ioctl(struct fuse_fs * fs,const char * path,unsigned int cmd,void * arg,struct fuse_file_info * fi,unsigned int flags,void * data)2308 int fuse_fs_ioctl(struct fuse_fs *fs, const char *path, unsigned int cmd,
2309 void *arg, struct fuse_file_info *fi, unsigned int flags,
2310 void *data)
2311 {
2312 fuse_get_context()->private_data = fs->user_data;
2313 if (fs->op.ioctl) {
2314 if (fs->debug)
2315 fuse_log(FUSE_LOG_DEBUG, "ioctl[%llu] 0x%x flags: 0x%x\n",
2316 (unsigned long long) fi->fh, cmd, flags);
2317
2318 return fs->op.ioctl(path, cmd, arg, fi, flags, data);
2319 } else
2320 return -ENOSYS;
2321 }
2322
fuse_fs_poll(struct fuse_fs * fs,const char * path,struct fuse_file_info * fi,struct fuse_pollhandle * ph,unsigned * reventsp)2323 int fuse_fs_poll(struct fuse_fs *fs, const char *path,
2324 struct fuse_file_info *fi, struct fuse_pollhandle *ph,
2325 unsigned *reventsp)
2326 {
2327 fuse_get_context()->private_data = fs->user_data;
2328 if (fs->op.poll) {
2329 int res;
2330
2331 if (fs->debug)
2332 fuse_log(FUSE_LOG_DEBUG, "poll[%llu] ph: %p, events 0x%x\n",
2333 (unsigned long long) fi->fh, ph,
2334 fi->poll_events);
2335
2336 res = fs->op.poll(path, fi, ph, reventsp);
2337
2338 if (fs->debug && !res)
2339 fuse_log(FUSE_LOG_DEBUG, " poll[%llu] revents: 0x%x\n",
2340 (unsigned long long) fi->fh, *reventsp);
2341
2342 return res;
2343 } else
2344 return -ENOSYS;
2345 }
2346
fuse_fs_fallocate(struct fuse_fs * fs,const char * path,int mode,off_t offset,off_t length,struct fuse_file_info * fi)2347 int fuse_fs_fallocate(struct fuse_fs *fs, const char *path, int mode,
2348 off_t offset, off_t length, struct fuse_file_info *fi)
2349 {
2350 fuse_get_context()->private_data = fs->user_data;
2351 if (fs->op.fallocate) {
2352 if (fs->debug)
2353 fuse_log(FUSE_LOG_DEBUG, "fallocate %s mode %x, offset: %llu, length: %llu\n",
2354 path,
2355 mode,
2356 (unsigned long long) offset,
2357 (unsigned long long) length);
2358
2359 return fs->op.fallocate(path, mode, offset, length, fi);
2360 } else
2361 return -ENOSYS;
2362 }
2363
fuse_fs_copy_file_range(struct fuse_fs * fs,const char * path_in,struct fuse_file_info * fi_in,off_t off_in,const char * path_out,struct fuse_file_info * fi_out,off_t off_out,size_t len,int flags)2364 ssize_t fuse_fs_copy_file_range(struct fuse_fs *fs, const char *path_in,
2365 struct fuse_file_info *fi_in, off_t off_in,
2366 const char *path_out,
2367 struct fuse_file_info *fi_out, off_t off_out,
2368 size_t len, int flags)
2369 {
2370 fuse_get_context()->private_data = fs->user_data;
2371 if (fs->op.copy_file_range) {
2372 if (fs->debug)
2373 fuse_log(FUSE_LOG_DEBUG, "copy_file_range from %s:%llu to "
2374 "%s:%llu, length: %llu\n",
2375 path_in,
2376 (unsigned long long) off_in,
2377 path_out,
2378 (unsigned long long) off_out,
2379 (unsigned long long) len);
2380
2381 return fs->op.copy_file_range(path_in, fi_in, off_in, path_out,
2382 fi_out, off_out, len, flags);
2383 } else
2384 return -ENOSYS;
2385 }
2386
fuse_fs_lseek(struct fuse_fs * fs,const char * path,off_t off,int whence,struct fuse_file_info * fi)2387 off_t fuse_fs_lseek(struct fuse_fs *fs, const char *path, off_t off, int whence,
2388 struct fuse_file_info *fi)
2389 {
2390 fuse_get_context()->private_data = fs->user_data;
2391 if (fs->op.lseek) {
2392 if (fs->debug) {
2393 char buf[10];
2394 fuse_log(FUSE_LOG_DEBUG, "lseek[%s] %llu %d\n",
2395 file_info_string(fi, buf, sizeof(buf)),
2396 (unsigned long long) off, whence);
2397 }
2398 return fs->op.lseek(path, off, whence, fi);
2399 } else {
2400 return -ENOSYS;
2401 }
2402 }
2403
is_open(struct fuse * f,fuse_ino_t dir,const char * name)2404 static int is_open(struct fuse *f, fuse_ino_t dir, const char *name)
2405 {
2406 struct node *node;
2407 int isopen = 0;
2408 pthread_mutex_lock(&f->lock);
2409 node = lookup_node(f, dir, name);
2410 if (node && node->open_count > 0)
2411 isopen = 1;
2412 pthread_mutex_unlock(&f->lock);
2413 return isopen;
2414 }
2415
hidden_name(struct fuse * f,fuse_ino_t dir,const char * oldname,char * newname,size_t bufsize)2416 static char *hidden_name(struct fuse *f, fuse_ino_t dir, const char *oldname,
2417 char *newname, size_t bufsize)
2418 {
2419 struct stat buf;
2420 struct node *node;
2421 struct node *newnode;
2422 char *newpath;
2423 int res;
2424 int failctr = 10;
2425
2426 do {
2427 pthread_mutex_lock(&f->lock);
2428 node = lookup_node(f, dir, oldname);
2429 if (node == NULL) {
2430 pthread_mutex_unlock(&f->lock);
2431 return NULL;
2432 }
2433 do {
2434 f->hidectr ++;
2435 snprintf(newname, bufsize, ".fuse_hidden%08x%08x",
2436 (unsigned int) node->nodeid, f->hidectr);
2437 newnode = lookup_node(f, dir, newname);
2438 } while(newnode);
2439
2440 res = try_get_path(f, dir, newname, &newpath, NULL, false);
2441 pthread_mutex_unlock(&f->lock);
2442 if (res)
2443 break;
2444
2445 memset(&buf, 0, sizeof(buf));
2446 res = fuse_fs_getattr(f->fs, newpath, &buf, NULL);
2447 if (res == -ENOENT)
2448 break;
2449 free(newpath);
2450 newpath = NULL;
2451 } while(res == 0 && --failctr);
2452
2453 return newpath;
2454 }
2455
hide_node(struct fuse * f,const char * oldpath,fuse_ino_t dir,const char * oldname)2456 static int hide_node(struct fuse *f, const char *oldpath,
2457 fuse_ino_t dir, const char *oldname)
2458 {
2459 char newname[64];
2460 char *newpath;
2461 int err = -EBUSY;
2462
2463 newpath = hidden_name(f, dir, oldname, newname, sizeof(newname));
2464 if (newpath) {
2465 err = fuse_fs_rename(f->fs, oldpath, newpath, 0);
2466 if (!err)
2467 err = rename_node(f, dir, oldname, dir, newname, 1);
2468 free(newpath);
2469 }
2470 return err;
2471 }
2472
mtime_eq(const struct stat * stbuf,const struct timespec * ts)2473 static int mtime_eq(const struct stat *stbuf, const struct timespec *ts)
2474 {
2475 return stbuf->st_mtime == ts->tv_sec &&
2476 ST_MTIM_NSEC(stbuf) == ts->tv_nsec;
2477 }
2478
2479 #ifndef CLOCK_MONOTONIC
2480 #define CLOCK_MONOTONIC CLOCK_REALTIME
2481 #endif
2482
curr_time(struct timespec * now)2483 static void curr_time(struct timespec *now)
2484 {
2485 static clockid_t clockid = CLOCK_MONOTONIC;
2486 int res = clock_gettime(clockid, now);
2487 if (res == -1 && errno == EINVAL) {
2488 clockid = CLOCK_REALTIME;
2489 res = clock_gettime(clockid, now);
2490 }
2491 if (res == -1) {
2492 perror("fuse: clock_gettime");
2493 abort();
2494 }
2495 }
2496
update_stat(struct node * node,const struct stat * stbuf)2497 static void update_stat(struct node *node, const struct stat *stbuf)
2498 {
2499 if (node->cache_valid && (!mtime_eq(stbuf, &node->mtime) ||
2500 stbuf->st_size != node->size))
2501 node->cache_valid = 0;
2502 node->mtime.tv_sec = stbuf->st_mtime;
2503 node->mtime.tv_nsec = ST_MTIM_NSEC(stbuf);
2504 node->size = stbuf->st_size;
2505 curr_time(&node->stat_updated);
2506 }
2507
do_lookup(struct fuse * f,fuse_ino_t nodeid,const char * name,struct fuse_entry_param * e)2508 static int do_lookup(struct fuse *f, fuse_ino_t nodeid, const char *name,
2509 struct fuse_entry_param *e)
2510 {
2511 struct node *node;
2512
2513 node = find_node(f, nodeid, name);
2514 if (node == NULL)
2515 return -ENOMEM;
2516
2517 e->ino = node->nodeid;
2518 e->generation = node->generation;
2519 e->entry_timeout = f->conf.entry_timeout;
2520 e->attr_timeout = f->conf.attr_timeout;
2521 if (f->conf.auto_cache) {
2522 pthread_mutex_lock(&f->lock);
2523 update_stat(node, &e->attr);
2524 pthread_mutex_unlock(&f->lock);
2525 }
2526 set_stat(f, e->ino, &e->attr);
2527 return 0;
2528 }
2529
lookup_path(struct fuse * f,fuse_ino_t nodeid,const char * name,const char * path,struct fuse_entry_param * e,struct fuse_file_info * fi)2530 static int lookup_path(struct fuse *f, fuse_ino_t nodeid,
2531 const char *name, const char *path,
2532 struct fuse_entry_param *e, struct fuse_file_info *fi)
2533 {
2534 int res;
2535
2536 memset(e, 0, sizeof(struct fuse_entry_param));
2537 res = fuse_fs_getattr(f->fs, path, &e->attr, fi);
2538 if (res == 0) {
2539 res = do_lookup(f, nodeid, name, e);
2540 if (res == 0 && f->conf.debug) {
2541 fuse_log(FUSE_LOG_DEBUG, " NODEID: %llu\n",
2542 (unsigned long long) e->ino);
2543 }
2544 }
2545 return res;
2546 }
2547
fuse_get_context_internal(void)2548 static struct fuse_context_i *fuse_get_context_internal(void)
2549 {
2550 return (struct fuse_context_i *) pthread_getspecific(fuse_context_key);
2551 }
2552
fuse_create_context(struct fuse * f)2553 static struct fuse_context_i *fuse_create_context(struct fuse *f)
2554 {
2555 struct fuse_context_i *c = fuse_get_context_internal();
2556 if (c == NULL) {
2557 c = (struct fuse_context_i *)
2558 calloc(1, sizeof(struct fuse_context_i));
2559 if (c == NULL) {
2560 /* This is hard to deal with properly, so just
2561 abort. If memory is so low that the
2562 context cannot be allocated, there's not
2563 much hope for the filesystem anyway */
2564 fuse_log(FUSE_LOG_ERR, "fuse: failed to allocate thread specific data\n");
2565 abort();
2566 }
2567 pthread_setspecific(fuse_context_key, c);
2568 } else {
2569 memset(c, 0, sizeof(*c));
2570 }
2571 c->ctx.fuse = f;
2572
2573 return c;
2574 }
2575
fuse_freecontext(void * data)2576 static void fuse_freecontext(void *data)
2577 {
2578 free(data);
2579 }
2580
fuse_create_context_key(void)2581 static int fuse_create_context_key(void)
2582 {
2583 int err = 0;
2584 pthread_mutex_lock(&fuse_context_lock);
2585 if (!fuse_context_ref) {
2586 err = pthread_key_create(&fuse_context_key, fuse_freecontext);
2587 if (err) {
2588 fuse_log(FUSE_LOG_ERR, "fuse: failed to create thread specific key: %s\n",
2589 strerror(err));
2590 pthread_mutex_unlock(&fuse_context_lock);
2591 return -1;
2592 }
2593 }
2594 fuse_context_ref++;
2595 pthread_mutex_unlock(&fuse_context_lock);
2596 return 0;
2597 }
2598
fuse_delete_context_key(void)2599 static void fuse_delete_context_key(void)
2600 {
2601 pthread_mutex_lock(&fuse_context_lock);
2602 fuse_context_ref--;
2603 if (!fuse_context_ref) {
2604 free(pthread_getspecific(fuse_context_key));
2605 pthread_key_delete(fuse_context_key);
2606 }
2607 pthread_mutex_unlock(&fuse_context_lock);
2608 }
2609
req_fuse_prepare(fuse_req_t req)2610 static struct fuse *req_fuse_prepare(fuse_req_t req)
2611 {
2612 struct fuse_context_i *c = fuse_create_context(req_fuse(req));
2613 const struct fuse_ctx *ctx = fuse_req_ctx(req);
2614 c->req = req;
2615 c->ctx.uid = ctx->uid;
2616 c->ctx.gid = ctx->gid;
2617 c->ctx.pid = ctx->pid;
2618 c->ctx.umask = ctx->umask;
2619 return c->ctx.fuse;
2620 }
2621
reply_err(fuse_req_t req,int err)2622 static inline void reply_err(fuse_req_t req, int err)
2623 {
2624 /* fuse_reply_err() uses non-negated errno values */
2625 fuse_reply_err(req, -err);
2626 }
2627
reply_entry(fuse_req_t req,const struct fuse_entry_param * e,int err)2628 static void reply_entry(fuse_req_t req, const struct fuse_entry_param *e,
2629 int err)
2630 {
2631 if (!err) {
2632 struct fuse *f = req_fuse(req);
2633 if (fuse_reply_entry(req, e) == -ENOENT) {
2634 /* Skip forget for negative result */
2635 if (e->ino != 0)
2636 forget_node(f, e->ino, 1);
2637 }
2638 } else
2639 reply_err(req, err);
2640 }
2641
fuse_fs_init(struct fuse_fs * fs,struct fuse_conn_info * conn,struct fuse_config * cfg)2642 void fuse_fs_init(struct fuse_fs *fs, struct fuse_conn_info *conn,
2643 struct fuse_config *cfg)
2644 {
2645 fuse_get_context()->private_data = fs->user_data;
2646 if (!fs->op.write_buf)
2647 conn->want &= ~FUSE_CAP_SPLICE_READ;
2648 if (!fs->op.lock)
2649 conn->want &= ~FUSE_CAP_POSIX_LOCKS;
2650 if (!fs->op.flock)
2651 conn->want &= ~FUSE_CAP_FLOCK_LOCKS;
2652 if (fs->op.init)
2653 fs->user_data = fs->op.init(conn, cfg);
2654 }
2655
fuse_lib_init(void * data,struct fuse_conn_info * conn)2656 static void fuse_lib_init(void *data, struct fuse_conn_info *conn)
2657 {
2658 struct fuse *f = (struct fuse *) data;
2659
2660 fuse_create_context(f);
2661 if(conn->capable & FUSE_CAP_EXPORT_SUPPORT)
2662 conn->want |= FUSE_CAP_EXPORT_SUPPORT;
2663 fuse_fs_init(f->fs, conn, &f->conf);
2664 }
2665
fuse_fs_destroy(struct fuse_fs * fs)2666 void fuse_fs_destroy(struct fuse_fs *fs)
2667 {
2668 fuse_get_context()->private_data = fs->user_data;
2669 if (fs->op.destroy)
2670 fs->op.destroy(fs->user_data);
2671 if (fs->m)
2672 fuse_put_module(fs->m);
2673 free(fs);
2674 }
2675
fuse_lib_destroy(void * data)2676 static void fuse_lib_destroy(void *data)
2677 {
2678 struct fuse *f = (struct fuse *) data;
2679
2680 fuse_create_context(f);
2681 fuse_fs_destroy(f->fs);
2682 f->fs = NULL;
2683 }
2684
fuse_lib_lookup(fuse_req_t req,fuse_ino_t parent,const char * name)2685 static void fuse_lib_lookup(fuse_req_t req, fuse_ino_t parent,
2686 const char *name)
2687 {
2688 struct fuse *f = req_fuse_prepare(req);
2689 struct fuse_entry_param e;
2690 char *path;
2691 int err;
2692 struct node *dot = NULL;
2693
2694 if (name[0] == '.') {
2695 int len = strlen(name);
2696
2697 if (len == 1 || (name[1] == '.' && len == 2)) {
2698 pthread_mutex_lock(&f->lock);
2699 if (len == 1) {
2700 if (f->conf.debug)
2701 fuse_log(FUSE_LOG_DEBUG, "LOOKUP-DOT\n");
2702 dot = get_node_nocheck(f, parent);
2703 if (dot == NULL) {
2704 pthread_mutex_unlock(&f->lock);
2705 reply_entry(req, &e, -ESTALE);
2706 return;
2707 }
2708 dot->refctr++;
2709 } else {
2710 if (f->conf.debug)
2711 fuse_log(FUSE_LOG_DEBUG, "LOOKUP-DOTDOT\n");
2712 parent = get_node(f, parent)->parent->nodeid;
2713 }
2714 pthread_mutex_unlock(&f->lock);
2715 name = NULL;
2716 }
2717 }
2718
2719 err = get_path_name(f, parent, name, &path);
2720 if (!err) {
2721 struct fuse_intr_data d;
2722 if (f->conf.debug)
2723 fuse_log(FUSE_LOG_DEBUG, "LOOKUP %s\n", path);
2724 fuse_prepare_interrupt(f, req, &d);
2725 err = lookup_path(f, parent, name, path, &e, NULL);
2726 if (err == -ENOENT && f->conf.negative_timeout != 0.0) {
2727 e.ino = 0;
2728 e.entry_timeout = f->conf.negative_timeout;
2729 err = 0;
2730 }
2731 fuse_finish_interrupt(f, req, &d);
2732 free_path(f, parent, path);
2733 }
2734 if (dot) {
2735 pthread_mutex_lock(&f->lock);
2736 unref_node(f, dot);
2737 pthread_mutex_unlock(&f->lock);
2738 }
2739 reply_entry(req, &e, err);
2740 }
2741
do_forget(struct fuse * f,fuse_ino_t ino,uint64_t nlookup)2742 static void do_forget(struct fuse *f, fuse_ino_t ino, uint64_t nlookup)
2743 {
2744 if (f->conf.debug)
2745 fuse_log(FUSE_LOG_DEBUG, "FORGET %llu/%llu\n", (unsigned long long)ino,
2746 (unsigned long long) nlookup);
2747 forget_node(f, ino, nlookup);
2748 }
2749
fuse_lib_forget(fuse_req_t req,fuse_ino_t ino,uint64_t nlookup)2750 static void fuse_lib_forget(fuse_req_t req, fuse_ino_t ino, uint64_t nlookup)
2751 {
2752 do_forget(req_fuse(req), ino, nlookup);
2753 fuse_reply_none(req);
2754 }
2755
fuse_lib_forget_multi(fuse_req_t req,size_t count,struct fuse_forget_data * forgets)2756 static void fuse_lib_forget_multi(fuse_req_t req, size_t count,
2757 struct fuse_forget_data *forgets)
2758 {
2759 struct fuse *f = req_fuse(req);
2760 size_t i;
2761
2762 for (i = 0; i < count; i++)
2763 do_forget(f, forgets[i].ino, forgets[i].nlookup);
2764
2765 fuse_reply_none(req);
2766 }
2767
2768
fuse_lib_getattr(fuse_req_t req,fuse_ino_t ino,struct fuse_file_info * fi)2769 static void fuse_lib_getattr(fuse_req_t req, fuse_ino_t ino,
2770 struct fuse_file_info *fi)
2771 {
2772 struct fuse *f = req_fuse_prepare(req);
2773 struct stat buf;
2774 char *path;
2775 int err;
2776
2777 memset(&buf, 0, sizeof(buf));
2778
2779 if (fi != NULL)
2780 err = get_path_nullok(f, ino, &path);
2781 else
2782 err = get_path(f, ino, &path);
2783 if (!err) {
2784 struct fuse_intr_data d;
2785 fuse_prepare_interrupt(f, req, &d);
2786 err = fuse_fs_getattr(f->fs, path, &buf, fi);
2787 fuse_finish_interrupt(f, req, &d);
2788 free_path(f, ino, path);
2789 }
2790 if (!err) {
2791 struct node *node;
2792
2793 pthread_mutex_lock(&f->lock);
2794 node = get_node(f, ino);
2795 if (node->is_hidden && buf.st_nlink > 0)
2796 buf.st_nlink--;
2797 if (f->conf.auto_cache)
2798 update_stat(node, &buf);
2799 pthread_mutex_unlock(&f->lock);
2800 set_stat(f, ino, &buf);
2801 fuse_reply_attr(req, &buf, f->conf.attr_timeout);
2802 } else
2803 reply_err(req, err);
2804 }
2805
fuse_fs_chmod(struct fuse_fs * fs,const char * path,mode_t mode,struct fuse_file_info * fi)2806 int fuse_fs_chmod(struct fuse_fs *fs, const char *path, mode_t mode,
2807 struct fuse_file_info *fi)
2808 {
2809 fuse_get_context()->private_data = fs->user_data;
2810 if (fs->op.chmod) {
2811 if (fs->debug) {
2812 char buf[10];
2813 fuse_log(FUSE_LOG_DEBUG, "chmod[%s] %s %llo\n",
2814 file_info_string(fi, buf, sizeof(buf)),
2815 path, (unsigned long long) mode);
2816 }
2817 return fs->op.chmod(path, mode, fi);
2818 }
2819 else
2820 return -ENOSYS;
2821 }
2822
fuse_lib_setattr(fuse_req_t req,fuse_ino_t ino,struct stat * attr,int valid,struct fuse_file_info * fi)2823 static void fuse_lib_setattr(fuse_req_t req, fuse_ino_t ino, struct stat *attr,
2824 int valid, struct fuse_file_info *fi)
2825 {
2826 struct fuse *f = req_fuse_prepare(req);
2827 struct stat buf;
2828 char *path;
2829 int err;
2830
2831 memset(&buf, 0, sizeof(buf));
2832 if (fi != NULL)
2833 err = get_path_nullok(f, ino, &path);
2834 else
2835 err = get_path(f, ino, &path);
2836 if (!err) {
2837 struct fuse_intr_data d;
2838 fuse_prepare_interrupt(f, req, &d);
2839 err = 0;
2840 if (!err && (valid & FUSE_SET_ATTR_MODE))
2841 err = fuse_fs_chmod(f->fs, path, attr->st_mode, fi);
2842 if (!err && (valid & (FUSE_SET_ATTR_UID | FUSE_SET_ATTR_GID))) {
2843 uid_t uid = (valid & FUSE_SET_ATTR_UID) ?
2844 attr->st_uid : (uid_t) -1;
2845 gid_t gid = (valid & FUSE_SET_ATTR_GID) ?
2846 attr->st_gid : (gid_t) -1;
2847 err = fuse_fs_chown(f->fs, path, uid, gid, fi);
2848 }
2849 if (!err && (valid & FUSE_SET_ATTR_SIZE)) {
2850 err = fuse_fs_truncate(f->fs, path,
2851 attr->st_size, fi);
2852 }
2853 #ifdef HAVE_UTIMENSAT
2854 if (!err &&
2855 (valid & (FUSE_SET_ATTR_ATIME | FUSE_SET_ATTR_MTIME))) {
2856 struct timespec tv[2];
2857
2858 tv[0].tv_sec = 0;
2859 tv[1].tv_sec = 0;
2860 tv[0].tv_nsec = UTIME_OMIT;
2861 tv[1].tv_nsec = UTIME_OMIT;
2862
2863 if (valid & FUSE_SET_ATTR_ATIME_NOW)
2864 tv[0].tv_nsec = UTIME_NOW;
2865 else if (valid & FUSE_SET_ATTR_ATIME)
2866 tv[0] = attr->st_atim;
2867
2868 if (valid & FUSE_SET_ATTR_MTIME_NOW)
2869 tv[1].tv_nsec = UTIME_NOW;
2870 else if (valid & FUSE_SET_ATTR_MTIME)
2871 tv[1] = attr->st_mtim;
2872
2873 err = fuse_fs_utimens(f->fs, path, tv, fi);
2874 } else
2875 #endif
2876 if (!err &&
2877 (valid & (FUSE_SET_ATTR_ATIME | FUSE_SET_ATTR_MTIME)) ==
2878 (FUSE_SET_ATTR_ATIME | FUSE_SET_ATTR_MTIME)) {
2879 struct timespec tv[2];
2880 tv[0].tv_sec = attr->st_atime;
2881 tv[0].tv_nsec = ST_ATIM_NSEC(attr);
2882 tv[1].tv_sec = attr->st_mtime;
2883 tv[1].tv_nsec = ST_MTIM_NSEC(attr);
2884 err = fuse_fs_utimens(f->fs, path, tv, fi);
2885 }
2886 if (!err) {
2887 err = fuse_fs_getattr(f->fs, path, &buf, fi);
2888 }
2889 fuse_finish_interrupt(f, req, &d);
2890 free_path(f, ino, path);
2891 }
2892 if (!err) {
2893 if (f->conf.auto_cache) {
2894 pthread_mutex_lock(&f->lock);
2895 update_stat(get_node(f, ino), &buf);
2896 pthread_mutex_unlock(&f->lock);
2897 }
2898 set_stat(f, ino, &buf);
2899 fuse_reply_attr(req, &buf, f->conf.attr_timeout);
2900 } else
2901 reply_err(req, err);
2902 }
2903
fuse_lib_access(fuse_req_t req,fuse_ino_t ino,int mask)2904 static void fuse_lib_access(fuse_req_t req, fuse_ino_t ino, int mask)
2905 {
2906 struct fuse *f = req_fuse_prepare(req);
2907 char *path;
2908 int err;
2909
2910 err = get_path(f, ino, &path);
2911 if (!err) {
2912 struct fuse_intr_data d;
2913
2914 fuse_prepare_interrupt(f, req, &d);
2915 err = fuse_fs_access(f->fs, path, mask);
2916 fuse_finish_interrupt(f, req, &d);
2917 free_path(f, ino, path);
2918 }
2919 reply_err(req, err);
2920 }
2921
fuse_lib_readlink(fuse_req_t req,fuse_ino_t ino)2922 static void fuse_lib_readlink(fuse_req_t req, fuse_ino_t ino)
2923 {
2924 struct fuse *f = req_fuse_prepare(req);
2925 char linkname[PATH_MAX + 1];
2926 char *path;
2927 int err;
2928
2929 err = get_path(f, ino, &path);
2930 if (!err) {
2931 struct fuse_intr_data d;
2932 fuse_prepare_interrupt(f, req, &d);
2933 err = fuse_fs_readlink(f->fs, path, linkname, sizeof(linkname));
2934 fuse_finish_interrupt(f, req, &d);
2935 free_path(f, ino, path);
2936 }
2937 if (!err) {
2938 linkname[PATH_MAX] = '\0';
2939 fuse_reply_readlink(req, linkname);
2940 } else
2941 reply_err(req, err);
2942 }
2943
fuse_lib_mknod(fuse_req_t req,fuse_ino_t parent,const char * name,mode_t mode,dev_t rdev)2944 static void fuse_lib_mknod(fuse_req_t req, fuse_ino_t parent, const char *name,
2945 mode_t mode, dev_t rdev)
2946 {
2947 struct fuse *f = req_fuse_prepare(req);
2948 struct fuse_entry_param e;
2949 char *path;
2950 int err;
2951
2952 err = get_path_name(f, parent, name, &path);
2953 if (!err) {
2954 struct fuse_intr_data d;
2955
2956 fuse_prepare_interrupt(f, req, &d);
2957 err = -ENOSYS;
2958 if (S_ISREG(mode)) {
2959 struct fuse_file_info fi;
2960
2961 memset(&fi, 0, sizeof(fi));
2962 fi.flags = O_CREAT | O_EXCL | O_WRONLY;
2963 err = fuse_fs_create(f->fs, path, mode, &fi);
2964 if (!err) {
2965 err = lookup_path(f, parent, name, path, &e,
2966 &fi);
2967 fuse_fs_release(f->fs, path, &fi);
2968 }
2969 }
2970 if (err == -ENOSYS) {
2971 err = fuse_fs_mknod(f->fs, path, mode, rdev);
2972 if (!err)
2973 err = lookup_path(f, parent, name, path, &e,
2974 NULL);
2975 }
2976 fuse_finish_interrupt(f, req, &d);
2977 free_path(f, parent, path);
2978 }
2979 reply_entry(req, &e, err);
2980 }
2981
fuse_lib_mkdir(fuse_req_t req,fuse_ino_t parent,const char * name,mode_t mode)2982 static void fuse_lib_mkdir(fuse_req_t req, fuse_ino_t parent, const char *name,
2983 mode_t mode)
2984 {
2985 struct fuse *f = req_fuse_prepare(req);
2986 struct fuse_entry_param e;
2987 char *path;
2988 int err;
2989
2990 err = get_path_name(f, parent, name, &path);
2991 if (!err) {
2992 struct fuse_intr_data d;
2993
2994 fuse_prepare_interrupt(f, req, &d);
2995 err = fuse_fs_mkdir(f->fs, path, mode);
2996 if (!err)
2997 err = lookup_path(f, parent, name, path, &e, NULL);
2998 fuse_finish_interrupt(f, req, &d);
2999 free_path(f, parent, path);
3000 }
3001 reply_entry(req, &e, err);
3002 }
3003
fuse_lib_unlink(fuse_req_t req,fuse_ino_t parent,const char * name)3004 static void fuse_lib_unlink(fuse_req_t req, fuse_ino_t parent,
3005 const char *name)
3006 {
3007 struct fuse *f = req_fuse_prepare(req);
3008 struct node *wnode;
3009 char *path;
3010 int err;
3011
3012 err = get_path_wrlock(f, parent, name, &path, &wnode);
3013 if (!err) {
3014 struct fuse_intr_data d;
3015
3016 fuse_prepare_interrupt(f, req, &d);
3017 if (!f->conf.hard_remove && is_open(f, parent, name)) {
3018 err = hide_node(f, path, parent, name);
3019 } else {
3020 err = fuse_fs_unlink(f->fs, path);
3021 if (!err)
3022 remove_node(f, parent, name);
3023 }
3024 fuse_finish_interrupt(f, req, &d);
3025 free_path_wrlock(f, parent, wnode, path);
3026 }
3027 reply_err(req, err);
3028 }
3029
fuse_lib_rmdir(fuse_req_t req,fuse_ino_t parent,const char * name)3030 static void fuse_lib_rmdir(fuse_req_t req, fuse_ino_t parent, const char *name)
3031 {
3032 struct fuse *f = req_fuse_prepare(req);
3033 struct node *wnode;
3034 char *path;
3035 int err;
3036
3037 err = get_path_wrlock(f, parent, name, &path, &wnode);
3038 if (!err) {
3039 struct fuse_intr_data d;
3040
3041 fuse_prepare_interrupt(f, req, &d);
3042 err = fuse_fs_rmdir(f->fs, path);
3043 fuse_finish_interrupt(f, req, &d);
3044 if (!err)
3045 remove_node(f, parent, name);
3046 free_path_wrlock(f, parent, wnode, path);
3047 }
3048 reply_err(req, err);
3049 }
3050
fuse_lib_symlink(fuse_req_t req,const char * linkname,fuse_ino_t parent,const char * name)3051 static void fuse_lib_symlink(fuse_req_t req, const char *linkname,
3052 fuse_ino_t parent, const char *name)
3053 {
3054 struct fuse *f = req_fuse_prepare(req);
3055 struct fuse_entry_param e;
3056 char *path;
3057 int err;
3058
3059 err = get_path_name(f, parent, name, &path);
3060 if (!err) {
3061 struct fuse_intr_data d;
3062
3063 fuse_prepare_interrupt(f, req, &d);
3064 err = fuse_fs_symlink(f->fs, linkname, path);
3065 if (!err)
3066 err = lookup_path(f, parent, name, path, &e, NULL);
3067 fuse_finish_interrupt(f, req, &d);
3068 free_path(f, parent, path);
3069 }
3070 reply_entry(req, &e, err);
3071 }
3072
fuse_lib_rename(fuse_req_t req,fuse_ino_t olddir,const char * oldname,fuse_ino_t newdir,const char * newname,unsigned int flags)3073 static void fuse_lib_rename(fuse_req_t req, fuse_ino_t olddir,
3074 const char *oldname, fuse_ino_t newdir,
3075 const char *newname, unsigned int flags)
3076 {
3077 struct fuse *f = req_fuse_prepare(req);
3078 char *oldpath;
3079 char *newpath;
3080 struct node *wnode1;
3081 struct node *wnode2;
3082 int err;
3083
3084 err = get_path2(f, olddir, oldname, newdir, newname,
3085 &oldpath, &newpath, &wnode1, &wnode2);
3086 if (!err) {
3087 struct fuse_intr_data d;
3088 err = 0;
3089 fuse_prepare_interrupt(f, req, &d);
3090 if (!f->conf.hard_remove && !(flags & RENAME_EXCHANGE) &&
3091 is_open(f, newdir, newname))
3092 err = hide_node(f, newpath, newdir, newname);
3093 if (!err) {
3094 err = fuse_fs_rename(f->fs, oldpath, newpath, flags);
3095 if (!err) {
3096 if (flags & RENAME_EXCHANGE) {
3097 err = exchange_node(f, olddir, oldname,
3098 newdir, newname);
3099 } else {
3100 err = rename_node(f, olddir, oldname,
3101 newdir, newname, 0);
3102 }
3103 }
3104 }
3105 fuse_finish_interrupt(f, req, &d);
3106 free_path2(f, olddir, newdir, wnode1, wnode2, oldpath, newpath);
3107 }
3108 reply_err(req, err);
3109 }
3110
fuse_lib_link(fuse_req_t req,fuse_ino_t ino,fuse_ino_t newparent,const char * newname)3111 static void fuse_lib_link(fuse_req_t req, fuse_ino_t ino, fuse_ino_t newparent,
3112 const char *newname)
3113 {
3114 struct fuse *f = req_fuse_prepare(req);
3115 struct fuse_entry_param e;
3116 char *oldpath;
3117 char *newpath;
3118 int err;
3119
3120 err = get_path2(f, ino, NULL, newparent, newname,
3121 &oldpath, &newpath, NULL, NULL);
3122 if (!err) {
3123 struct fuse_intr_data d;
3124
3125 fuse_prepare_interrupt(f, req, &d);
3126 err = fuse_fs_link(f->fs, oldpath, newpath);
3127 if (!err)
3128 err = lookup_path(f, newparent, newname, newpath,
3129 &e, NULL);
3130 fuse_finish_interrupt(f, req, &d);
3131 free_path2(f, ino, newparent, NULL, NULL, oldpath, newpath);
3132 }
3133 reply_entry(req, &e, err);
3134 }
3135
fuse_do_release(struct fuse * f,fuse_ino_t ino,const char * path,struct fuse_file_info * fi)3136 static void fuse_do_release(struct fuse *f, fuse_ino_t ino, const char *path,
3137 struct fuse_file_info *fi)
3138 {
3139 struct node *node;
3140 int unlink_hidden = 0;
3141
3142 fuse_fs_release(f->fs, path, fi);
3143
3144 pthread_mutex_lock(&f->lock);
3145 node = get_node(f, ino);
3146 assert(node->open_count > 0);
3147 --node->open_count;
3148 if (node->is_hidden && !node->open_count) {
3149 unlink_hidden = 1;
3150 node->is_hidden = 0;
3151 }
3152 pthread_mutex_unlock(&f->lock);
3153
3154 if(unlink_hidden) {
3155 if (path) {
3156 fuse_fs_unlink(f->fs, path);
3157 } else if (f->conf.nullpath_ok) {
3158 char *unlinkpath;
3159
3160 if (get_path(f, ino, &unlinkpath) == 0)
3161 fuse_fs_unlink(f->fs, unlinkpath);
3162
3163 free_path(f, ino, unlinkpath);
3164 }
3165 }
3166 }
3167
fuse_lib_create(fuse_req_t req,fuse_ino_t parent,const char * name,mode_t mode,struct fuse_file_info * fi)3168 static void fuse_lib_create(fuse_req_t req, fuse_ino_t parent,
3169 const char *name, mode_t mode,
3170 struct fuse_file_info *fi)
3171 {
3172 struct fuse *f = req_fuse_prepare(req);
3173 struct fuse_intr_data d;
3174 struct fuse_entry_param e;
3175 char *path;
3176 int err;
3177
3178 err = get_path_name(f, parent, name, &path);
3179 if (!err) {
3180 fuse_prepare_interrupt(f, req, &d);
3181 err = fuse_fs_create(f->fs, path, mode, fi);
3182 if (!err) {
3183 err = lookup_path(f, parent, name, path, &e, fi);
3184 if (err)
3185 fuse_fs_release(f->fs, path, fi);
3186 else if (!S_ISREG(e.attr.st_mode)) {
3187 err = -EIO;
3188 fuse_fs_release(f->fs, path, fi);
3189 forget_node(f, e.ino, 1);
3190 } else {
3191 if (f->conf.direct_io)
3192 fi->direct_io = 1;
3193 if (f->conf.kernel_cache)
3194 fi->keep_cache = 1;
3195
3196 }
3197 }
3198 fuse_finish_interrupt(f, req, &d);
3199 }
3200 if (!err) {
3201 pthread_mutex_lock(&f->lock);
3202 get_node(f, e.ino)->open_count++;
3203 pthread_mutex_unlock(&f->lock);
3204 if (fuse_reply_create(req, &e, fi) == -ENOENT) {
3205 /* The open syscall was interrupted, so it
3206 must be cancelled */
3207 fuse_do_release(f, e.ino, path, fi);
3208 forget_node(f, e.ino, 1);
3209 }
3210 } else {
3211 reply_err(req, err);
3212 }
3213
3214 free_path(f, parent, path);
3215 }
3216
diff_timespec(const struct timespec * t1,const struct timespec * t2)3217 static double diff_timespec(const struct timespec *t1,
3218 const struct timespec *t2)
3219 {
3220 return (t1->tv_sec - t2->tv_sec) +
3221 ((double) t1->tv_nsec - (double) t2->tv_nsec) / 1000000000.0;
3222 }
3223
open_auto_cache(struct fuse * f,fuse_ino_t ino,const char * path,struct fuse_file_info * fi)3224 static void open_auto_cache(struct fuse *f, fuse_ino_t ino, const char *path,
3225 struct fuse_file_info *fi)
3226 {
3227 struct node *node;
3228
3229 pthread_mutex_lock(&f->lock);
3230 node = get_node(f, ino);
3231 if (node->cache_valid) {
3232 struct timespec now;
3233
3234 curr_time(&now);
3235 if (diff_timespec(&now, &node->stat_updated) >
3236 f->conf.ac_attr_timeout) {
3237 struct stat stbuf;
3238 int err;
3239 pthread_mutex_unlock(&f->lock);
3240 err = fuse_fs_getattr(f->fs, path, &stbuf, fi);
3241 pthread_mutex_lock(&f->lock);
3242 if (!err)
3243 update_stat(node, &stbuf);
3244 else
3245 node->cache_valid = 0;
3246 }
3247 }
3248 if (node->cache_valid)
3249 fi->keep_cache = 1;
3250
3251 node->cache_valid = 1;
3252 pthread_mutex_unlock(&f->lock);
3253 }
3254
fuse_lib_open(fuse_req_t req,fuse_ino_t ino,struct fuse_file_info * fi)3255 static void fuse_lib_open(fuse_req_t req, fuse_ino_t ino,
3256 struct fuse_file_info *fi)
3257 {
3258 struct fuse *f = req_fuse_prepare(req);
3259 struct fuse_intr_data d;
3260 char *path;
3261 int err;
3262
3263 err = get_path(f, ino, &path);
3264 if (!err) {
3265 fuse_prepare_interrupt(f, req, &d);
3266 err = fuse_fs_open(f->fs, path, fi);
3267 if (!err) {
3268 if (f->conf.direct_io)
3269 fi->direct_io = 1;
3270 if (f->conf.kernel_cache)
3271 fi->keep_cache = 1;
3272
3273 if (f->conf.auto_cache)
3274 open_auto_cache(f, ino, path, fi);
3275 }
3276 fuse_finish_interrupt(f, req, &d);
3277 }
3278 if (!err) {
3279 pthread_mutex_lock(&f->lock);
3280 get_node(f, ino)->open_count++;
3281 pthread_mutex_unlock(&f->lock);
3282 if (fuse_reply_open(req, fi) == -ENOENT) {
3283 /* The open syscall was interrupted, so it
3284 must be cancelled */
3285 fuse_do_release(f, ino, path, fi);
3286 }
3287 } else
3288 reply_err(req, err);
3289
3290 free_path(f, ino, path);
3291 }
3292
fuse_lib_read(fuse_req_t req,fuse_ino_t ino,size_t size,off_t off,struct fuse_file_info * fi)3293 static void fuse_lib_read(fuse_req_t req, fuse_ino_t ino, size_t size,
3294 off_t off, struct fuse_file_info *fi)
3295 {
3296 struct fuse *f = req_fuse_prepare(req);
3297 struct fuse_bufvec *buf = NULL;
3298 char *path;
3299 int res;
3300
3301 res = get_path_nullok(f, ino, &path);
3302 if (res == 0) {
3303 struct fuse_intr_data d;
3304
3305 fuse_prepare_interrupt(f, req, &d);
3306 res = fuse_fs_read_buf(f->fs, path, &buf, size, off, fi);
3307 fuse_finish_interrupt(f, req, &d);
3308 free_path(f, ino, path);
3309 }
3310
3311 if (res == 0)
3312 fuse_reply_data(req, buf, FUSE_BUF_SPLICE_MOVE);
3313 else
3314 reply_err(req, res);
3315
3316 fuse_free_buf(buf);
3317 }
3318
fuse_lib_write_buf(fuse_req_t req,fuse_ino_t ino,struct fuse_bufvec * buf,off_t off,struct fuse_file_info * fi)3319 static void fuse_lib_write_buf(fuse_req_t req, fuse_ino_t ino,
3320 struct fuse_bufvec *buf, off_t off,
3321 struct fuse_file_info *fi)
3322 {
3323 struct fuse *f = req_fuse_prepare(req);
3324 char *path;
3325 int res;
3326
3327 res = get_path_nullok(f, ino, &path);
3328 if (res == 0) {
3329 struct fuse_intr_data d;
3330
3331 fuse_prepare_interrupt(f, req, &d);
3332 res = fuse_fs_write_buf(f->fs, path, buf, off, fi);
3333 fuse_finish_interrupt(f, req, &d);
3334 free_path(f, ino, path);
3335 }
3336
3337 if (res >= 0)
3338 fuse_reply_write(req, res);
3339 else
3340 reply_err(req, res);
3341 }
3342
fuse_lib_fsync(fuse_req_t req,fuse_ino_t ino,int datasync,struct fuse_file_info * fi)3343 static void fuse_lib_fsync(fuse_req_t req, fuse_ino_t ino, int datasync,
3344 struct fuse_file_info *fi)
3345 {
3346 struct fuse *f = req_fuse_prepare(req);
3347 char *path;
3348 int err;
3349
3350 err = get_path_nullok(f, ino, &path);
3351 if (!err) {
3352 struct fuse_intr_data d;
3353
3354 fuse_prepare_interrupt(f, req, &d);
3355 err = fuse_fs_fsync(f->fs, path, datasync, fi);
3356 fuse_finish_interrupt(f, req, &d);
3357 free_path(f, ino, path);
3358 }
3359 reply_err(req, err);
3360 }
3361
get_dirhandle(const struct fuse_file_info * llfi,struct fuse_file_info * fi)3362 static struct fuse_dh *get_dirhandle(const struct fuse_file_info *llfi,
3363 struct fuse_file_info *fi)
3364 {
3365 struct fuse_dh *dh = (struct fuse_dh *) (uintptr_t) llfi->fh;
3366 memset(fi, 0, sizeof(struct fuse_file_info));
3367 fi->fh = dh->fh;
3368 return dh;
3369 }
3370
fuse_lib_opendir(fuse_req_t req,fuse_ino_t ino,struct fuse_file_info * llfi)3371 static void fuse_lib_opendir(fuse_req_t req, fuse_ino_t ino,
3372 struct fuse_file_info *llfi)
3373 {
3374 struct fuse *f = req_fuse_prepare(req);
3375 struct fuse_intr_data d;
3376 struct fuse_dh *dh;
3377 struct fuse_file_info fi;
3378 char *path;
3379 int err;
3380
3381 dh = (struct fuse_dh *) malloc(sizeof(struct fuse_dh));
3382 if (dh == NULL) {
3383 reply_err(req, -ENOMEM);
3384 return;
3385 }
3386 memset(dh, 0, sizeof(struct fuse_dh));
3387 dh->fuse = f;
3388 dh->contents = NULL;
3389 dh->first = NULL;
3390 dh->len = 0;
3391 dh->filled = 0;
3392 dh->nodeid = ino;
3393 fuse_mutex_init(&dh->lock);
3394
3395 llfi->fh = (uintptr_t) dh;
3396
3397 memset(&fi, 0, sizeof(fi));
3398 fi.flags = llfi->flags;
3399
3400 err = get_path(f, ino, &path);
3401 if (!err) {
3402 fuse_prepare_interrupt(f, req, &d);
3403 err = fuse_fs_opendir(f->fs, path, &fi);
3404 fuse_finish_interrupt(f, req, &d);
3405 dh->fh = fi.fh;
3406 }
3407 if (!err) {
3408 if (fuse_reply_open(req, llfi) == -ENOENT) {
3409 /* The opendir syscall was interrupted, so it
3410 must be cancelled */
3411 fuse_fs_releasedir(f->fs, path, &fi);
3412 pthread_mutex_destroy(&dh->lock);
3413 free(dh);
3414 }
3415 } else {
3416 reply_err(req, err);
3417 pthread_mutex_destroy(&dh->lock);
3418 free(dh);
3419 }
3420 free_path(f, ino, path);
3421 }
3422
extend_contents(struct fuse_dh * dh,unsigned minsize)3423 static int extend_contents(struct fuse_dh *dh, unsigned minsize)
3424 {
3425 if (minsize > dh->size) {
3426 char *newptr;
3427 unsigned newsize = dh->size;
3428 if (!newsize)
3429 newsize = 1024;
3430 while (newsize < minsize) {
3431 if (newsize >= 0x80000000)
3432 newsize = 0xffffffff;
3433 else
3434 newsize *= 2;
3435 }
3436
3437 newptr = (char *) realloc(dh->contents, newsize);
3438 if (!newptr) {
3439 dh->error = -ENOMEM;
3440 return -1;
3441 }
3442 dh->contents = newptr;
3443 dh->size = newsize;
3444 }
3445 return 0;
3446 }
3447
fuse_add_direntry_to_dh(struct fuse_dh * dh,const char * name,struct stat * st)3448 static int fuse_add_direntry_to_dh(struct fuse_dh *dh, const char *name,
3449 struct stat *st)
3450 {
3451 struct fuse_direntry *de;
3452
3453 de = malloc(sizeof(struct fuse_direntry));
3454 if (!de) {
3455 dh->error = -ENOMEM;
3456 return -1;
3457 }
3458 de->name = strdup(name);
3459 if (!de->name) {
3460 dh->error = -ENOMEM;
3461 free(de);
3462 return -1;
3463 }
3464 de->stat = *st;
3465 de->next = NULL;
3466
3467 *dh->last = de;
3468 dh->last = &de->next;
3469
3470 return 0;
3471 }
3472
lookup_nodeid(struct fuse * f,fuse_ino_t parent,const char * name)3473 static fuse_ino_t lookup_nodeid(struct fuse *f, fuse_ino_t parent,
3474 const char *name)
3475 {
3476 struct node *node;
3477 fuse_ino_t res = FUSE_UNKNOWN_INO;
3478
3479 pthread_mutex_lock(&f->lock);
3480 node = lookup_node(f, parent, name);
3481 if (node)
3482 res = node->nodeid;
3483 pthread_mutex_unlock(&f->lock);
3484
3485 return res;
3486 }
3487
fill_dir(void * dh_,const char * name,const struct stat * statp,off_t off,enum fuse_fill_dir_flags flags)3488 static int fill_dir(void *dh_, const char *name, const struct stat *statp,
3489 off_t off, enum fuse_fill_dir_flags flags)
3490 {
3491 struct fuse_dh *dh = (struct fuse_dh *) dh_;
3492 struct stat stbuf;
3493
3494 if ((flags & ~FUSE_FILL_DIR_PLUS) != 0) {
3495 dh->error = -EIO;
3496 return 1;
3497 }
3498
3499 if (statp)
3500 stbuf = *statp;
3501 else {
3502 memset(&stbuf, 0, sizeof(stbuf));
3503 stbuf.st_ino = FUSE_UNKNOWN_INO;
3504 }
3505
3506 if (!dh->fuse->conf.use_ino) {
3507 stbuf.st_ino = FUSE_UNKNOWN_INO;
3508 if (dh->fuse->conf.readdir_ino) {
3509 stbuf.st_ino = (ino_t)
3510 lookup_nodeid(dh->fuse, dh->nodeid, name);
3511 }
3512 }
3513
3514 if (off) {
3515 size_t newlen;
3516
3517 if (dh->filled) {
3518 dh->error = -EIO;
3519 return 1;
3520 }
3521
3522 if (dh->first) {
3523 dh->error = -EIO;
3524 return 1;
3525 }
3526
3527 if (extend_contents(dh, dh->needlen) == -1)
3528 return 1;
3529
3530 newlen = dh->len +
3531 fuse_add_direntry(dh->req, dh->contents + dh->len,
3532 dh->needlen - dh->len, name,
3533 &stbuf, off);
3534 if (newlen > dh->needlen)
3535 return 1;
3536
3537 dh->len = newlen;
3538 } else {
3539 dh->filled = 1;
3540
3541 if (fuse_add_direntry_to_dh(dh, name, &stbuf) == -1)
3542 return 1;
3543 }
3544 return 0;
3545 }
3546
is_dot_or_dotdot(const char * name)3547 static int is_dot_or_dotdot(const char *name)
3548 {
3549 return name[0] == '.' && (name[1] == '\0' ||
3550 (name[1] == '.' && name[2] == '\0'));
3551 }
3552
fill_dir_plus(void * dh_,const char * name,const struct stat * statp,off_t off,enum fuse_fill_dir_flags flags)3553 static int fill_dir_plus(void *dh_, const char *name, const struct stat *statp,
3554 off_t off, enum fuse_fill_dir_flags flags)
3555 {
3556 struct fuse_dh *dh = (struct fuse_dh *) dh_;
3557 struct fuse_entry_param e = {
3558 /* ino=0 tells the kernel to ignore readdirplus stat info */
3559 .ino = 0,
3560 };
3561 struct fuse *f = dh->fuse;
3562 int res;
3563
3564 if ((flags & ~FUSE_FILL_DIR_PLUS) != 0) {
3565 dh->error = -EIO;
3566 return 1;
3567 }
3568
3569 if (off && statp && (flags & FUSE_FILL_DIR_PLUS)) {
3570 e.attr = *statp;
3571
3572 if (!is_dot_or_dotdot(name)) {
3573 res = do_lookup(f, dh->nodeid, name, &e);
3574 if (res) {
3575 dh->error = res;
3576 return 1;
3577 }
3578 }
3579 } else {
3580 e.attr.st_ino = FUSE_UNKNOWN_INO;
3581 if (!f->conf.use_ino && f->conf.readdir_ino) {
3582 e.attr.st_ino = (ino_t)
3583 lookup_nodeid(f, dh->nodeid, name);
3584 }
3585 }
3586
3587 if (off) {
3588 size_t newlen;
3589
3590 if (dh->filled) {
3591 dh->error = -EIO;
3592 return 1;
3593 }
3594
3595 if (dh->first) {
3596 dh->error = -EIO;
3597 return 1;
3598 }
3599 if (extend_contents(dh, dh->needlen) == -1)
3600 return 1;
3601
3602 newlen = dh->len +
3603 fuse_add_direntry_plus(dh->req, dh->contents + dh->len,
3604 dh->needlen - dh->len, name,
3605 &e, off);
3606 if (newlen > dh->needlen)
3607 return 1;
3608 dh->len = newlen;
3609 } else {
3610 dh->filled = 1;
3611
3612 if (fuse_add_direntry_to_dh(dh, name, &e.attr) == -1)
3613 return 1;
3614 }
3615
3616 return 0;
3617 }
3618
free_direntries(struct fuse_direntry * de)3619 static void free_direntries(struct fuse_direntry *de)
3620 {
3621 while (de) {
3622 struct fuse_direntry *next = de->next;
3623 free(de->name);
3624 free(de);
3625 de = next;
3626 }
3627 }
3628
readdir_fill(struct fuse * f,fuse_req_t req,fuse_ino_t ino,size_t size,off_t off,struct fuse_dh * dh,struct fuse_file_info * fi,enum fuse_readdir_flags flags)3629 static int readdir_fill(struct fuse *f, fuse_req_t req, fuse_ino_t ino,
3630 size_t size, off_t off, struct fuse_dh *dh,
3631 struct fuse_file_info *fi,
3632 enum fuse_readdir_flags flags)
3633 {
3634 char *path;
3635 int err;
3636
3637 if (f->fs->op.readdir)
3638 err = get_path_nullok(f, ino, &path);
3639 else
3640 err = get_path(f, ino, &path);
3641 if (!err) {
3642 struct fuse_intr_data d;
3643 fuse_fill_dir_t filler = fill_dir;
3644
3645 if (flags & FUSE_READDIR_PLUS)
3646 filler = fill_dir_plus;
3647
3648 free_direntries(dh->first);
3649 dh->first = NULL;
3650 dh->last = &dh->first;
3651 dh->len = 0;
3652 dh->error = 0;
3653 dh->needlen = size;
3654 dh->filled = 0;
3655 dh->req = req;
3656 fuse_prepare_interrupt(f, req, &d);
3657 err = fuse_fs_readdir(f->fs, path, dh, filler, off, fi, flags);
3658 fuse_finish_interrupt(f, req, &d);
3659 dh->req = NULL;
3660 if (!err)
3661 err = dh->error;
3662 if (err)
3663 dh->filled = 0;
3664 free_path(f, ino, path);
3665 }
3666 return err;
3667 }
3668
readdir_fill_from_list(fuse_req_t req,struct fuse_dh * dh,off_t off,enum fuse_readdir_flags flags)3669 static int readdir_fill_from_list(fuse_req_t req, struct fuse_dh *dh,
3670 off_t off, enum fuse_readdir_flags flags)
3671 {
3672 off_t pos;
3673 struct fuse_direntry *de = dh->first;
3674
3675 dh->len = 0;
3676
3677 if (extend_contents(dh, dh->needlen) == -1)
3678 return dh->error;
3679
3680 for (pos = 0; pos < off; pos++) {
3681 if (!de)
3682 break;
3683
3684 de = de->next;
3685 }
3686 while (de) {
3687 char *p = dh->contents + dh->len;
3688 unsigned rem = dh->needlen - dh->len;
3689 unsigned thislen;
3690 unsigned newlen;
3691 pos++;
3692
3693 if (flags & FUSE_READDIR_PLUS) {
3694 struct fuse_entry_param e = {
3695 .ino = 0,
3696 .attr = de->stat,
3697 };
3698 thislen = fuse_add_direntry_plus(req, p, rem,
3699 de->name, &e, pos);
3700 } else {
3701 thislen = fuse_add_direntry(req, p, rem,
3702 de->name, &de->stat, pos);
3703 }
3704 newlen = dh->len + thislen;
3705 if (newlen > dh->needlen)
3706 break;
3707 dh->len = newlen;
3708 de = de->next;
3709 }
3710 return 0;
3711 }
3712
fuse_readdir_common(fuse_req_t req,fuse_ino_t ino,size_t size,off_t off,struct fuse_file_info * llfi,enum fuse_readdir_flags flags)3713 static void fuse_readdir_common(fuse_req_t req, fuse_ino_t ino, size_t size,
3714 off_t off, struct fuse_file_info *llfi,
3715 enum fuse_readdir_flags flags)
3716 {
3717 struct fuse *f = req_fuse_prepare(req);
3718 struct fuse_file_info fi;
3719 struct fuse_dh *dh = get_dirhandle(llfi, &fi);
3720 int err;
3721
3722 pthread_mutex_lock(&dh->lock);
3723 /* According to SUS, directory contents need to be refreshed on
3724 rewinddir() */
3725 if (!off)
3726 dh->filled = 0;
3727
3728 if (!dh->filled) {
3729 err = readdir_fill(f, req, ino, size, off, dh, &fi, flags);
3730 if (err) {
3731 reply_err(req, err);
3732 goto out;
3733 }
3734 }
3735 if (dh->filled) {
3736 dh->needlen = size;
3737 err = readdir_fill_from_list(req, dh, off, flags);
3738 if (err) {
3739 reply_err(req, err);
3740 goto out;
3741 }
3742 }
3743 fuse_reply_buf(req, dh->contents, dh->len);
3744 out:
3745 pthread_mutex_unlock(&dh->lock);
3746 }
3747
fuse_lib_readdir(fuse_req_t req,fuse_ino_t ino,size_t size,off_t off,struct fuse_file_info * llfi)3748 static void fuse_lib_readdir(fuse_req_t req, fuse_ino_t ino, size_t size,
3749 off_t off, struct fuse_file_info *llfi)
3750 {
3751 fuse_readdir_common(req, ino, size, off, llfi, 0);
3752 }
3753
fuse_lib_readdirplus(fuse_req_t req,fuse_ino_t ino,size_t size,off_t off,struct fuse_file_info * llfi)3754 static void fuse_lib_readdirplus(fuse_req_t req, fuse_ino_t ino, size_t size,
3755 off_t off, struct fuse_file_info *llfi)
3756 {
3757 fuse_readdir_common(req, ino, size, off, llfi, FUSE_READDIR_PLUS);
3758 }
3759
fuse_lib_releasedir(fuse_req_t req,fuse_ino_t ino,struct fuse_file_info * llfi)3760 static void fuse_lib_releasedir(fuse_req_t req, fuse_ino_t ino,
3761 struct fuse_file_info *llfi)
3762 {
3763 struct fuse *f = req_fuse_prepare(req);
3764 struct fuse_intr_data d;
3765 struct fuse_file_info fi;
3766 struct fuse_dh *dh = get_dirhandle(llfi, &fi);
3767 char *path;
3768
3769 get_path_nullok(f, ino, &path);
3770
3771 fuse_prepare_interrupt(f, req, &d);
3772 fuse_fs_releasedir(f->fs, path, &fi);
3773 fuse_finish_interrupt(f, req, &d);
3774 free_path(f, ino, path);
3775
3776 pthread_mutex_lock(&dh->lock);
3777 pthread_mutex_unlock(&dh->lock);
3778 pthread_mutex_destroy(&dh->lock);
3779 free_direntries(dh->first);
3780 free(dh->contents);
3781 free(dh);
3782 reply_err(req, 0);
3783 }
3784
fuse_lib_fsyncdir(fuse_req_t req,fuse_ino_t ino,int datasync,struct fuse_file_info * llfi)3785 static void fuse_lib_fsyncdir(fuse_req_t req, fuse_ino_t ino, int datasync,
3786 struct fuse_file_info *llfi)
3787 {
3788 struct fuse *f = req_fuse_prepare(req);
3789 struct fuse_file_info fi;
3790 char *path;
3791 int err;
3792
3793 get_dirhandle(llfi, &fi);
3794
3795 err = get_path_nullok(f, ino, &path);
3796 if (!err) {
3797 struct fuse_intr_data d;
3798 fuse_prepare_interrupt(f, req, &d);
3799 err = fuse_fs_fsyncdir(f->fs, path, datasync, &fi);
3800 fuse_finish_interrupt(f, req, &d);
3801 free_path(f, ino, path);
3802 }
3803 reply_err(req, err);
3804 }
3805
fuse_lib_statfs(fuse_req_t req,fuse_ino_t ino)3806 static void fuse_lib_statfs(fuse_req_t req, fuse_ino_t ino)
3807 {
3808 struct fuse *f = req_fuse_prepare(req);
3809 struct statvfs buf;
3810 char *path = NULL;
3811 int err = 0;
3812
3813 memset(&buf, 0, sizeof(buf));
3814 if (ino)
3815 err = get_path(f, ino, &path);
3816
3817 if (!err) {
3818 struct fuse_intr_data d;
3819 fuse_prepare_interrupt(f, req, &d);
3820 err = fuse_fs_statfs(f->fs, path ? path : "/", &buf);
3821 fuse_finish_interrupt(f, req, &d);
3822 free_path(f, ino, path);
3823 }
3824
3825 if (!err)
3826 fuse_reply_statfs(req, &buf);
3827 else
3828 reply_err(req, err);
3829 }
3830
fuse_lib_setxattr(fuse_req_t req,fuse_ino_t ino,const char * name,const char * value,size_t size,int flags)3831 static void fuse_lib_setxattr(fuse_req_t req, fuse_ino_t ino, const char *name,
3832 const char *value, size_t size, int flags)
3833 {
3834 struct fuse *f = req_fuse_prepare(req);
3835 char *path;
3836 int err;
3837
3838 err = get_path(f, ino, &path);
3839 if (!err) {
3840 struct fuse_intr_data d;
3841 fuse_prepare_interrupt(f, req, &d);
3842 err = fuse_fs_setxattr(f->fs, path, name, value, size, flags);
3843 fuse_finish_interrupt(f, req, &d);
3844 free_path(f, ino, path);
3845 }
3846 reply_err(req, err);
3847 }
3848
common_getxattr(struct fuse * f,fuse_req_t req,fuse_ino_t ino,const char * name,char * value,size_t size)3849 static int common_getxattr(struct fuse *f, fuse_req_t req, fuse_ino_t ino,
3850 const char *name, char *value, size_t size)
3851 {
3852 int err;
3853 char *path;
3854
3855 err = get_path(f, ino, &path);
3856 if (!err) {
3857 struct fuse_intr_data d;
3858 fuse_prepare_interrupt(f, req, &d);
3859 err = fuse_fs_getxattr(f->fs, path, name, value, size);
3860 fuse_finish_interrupt(f, req, &d);
3861 free_path(f, ino, path);
3862 }
3863 return err;
3864 }
3865
fuse_lib_getxattr(fuse_req_t req,fuse_ino_t ino,const char * name,size_t size)3866 static void fuse_lib_getxattr(fuse_req_t req, fuse_ino_t ino, const char *name,
3867 size_t size)
3868 {
3869 struct fuse *f = req_fuse_prepare(req);
3870 int res;
3871
3872 if (size) {
3873 char *value = (char *) malloc(size);
3874 if (value == NULL) {
3875 reply_err(req, -ENOMEM);
3876 return;
3877 }
3878 res = common_getxattr(f, req, ino, name, value, size);
3879 if (res > 0)
3880 fuse_reply_buf(req, value, res);
3881 else
3882 reply_err(req, res);
3883 free(value);
3884 } else {
3885 res = common_getxattr(f, req, ino, name, NULL, 0);
3886 if (res >= 0)
3887 fuse_reply_xattr(req, res);
3888 else
3889 reply_err(req, res);
3890 }
3891 }
3892
common_listxattr(struct fuse * f,fuse_req_t req,fuse_ino_t ino,char * list,size_t size)3893 static int common_listxattr(struct fuse *f, fuse_req_t req, fuse_ino_t ino,
3894 char *list, size_t size)
3895 {
3896 char *path;
3897 int err;
3898
3899 err = get_path(f, ino, &path);
3900 if (!err) {
3901 struct fuse_intr_data d;
3902 fuse_prepare_interrupt(f, req, &d);
3903 err = fuse_fs_listxattr(f->fs, path, list, size);
3904 fuse_finish_interrupt(f, req, &d);
3905 free_path(f, ino, path);
3906 }
3907 return err;
3908 }
3909
fuse_lib_listxattr(fuse_req_t req,fuse_ino_t ino,size_t size)3910 static void fuse_lib_listxattr(fuse_req_t req, fuse_ino_t ino, size_t size)
3911 {
3912 struct fuse *f = req_fuse_prepare(req);
3913 int res;
3914
3915 if (size) {
3916 char *list = (char *) malloc(size);
3917 if (list == NULL) {
3918 reply_err(req, -ENOMEM);
3919 return;
3920 }
3921 res = common_listxattr(f, req, ino, list, size);
3922 if (res > 0)
3923 fuse_reply_buf(req, list, res);
3924 else
3925 reply_err(req, res);
3926 free(list);
3927 } else {
3928 res = common_listxattr(f, req, ino, NULL, 0);
3929 if (res >= 0)
3930 fuse_reply_xattr(req, res);
3931 else
3932 reply_err(req, res);
3933 }
3934 }
3935
fuse_lib_removexattr(fuse_req_t req,fuse_ino_t ino,const char * name)3936 static void fuse_lib_removexattr(fuse_req_t req, fuse_ino_t ino,
3937 const char *name)
3938 {
3939 struct fuse *f = req_fuse_prepare(req);
3940 char *path;
3941 int err;
3942
3943 err = get_path(f, ino, &path);
3944 if (!err) {
3945 struct fuse_intr_data d;
3946 fuse_prepare_interrupt(f, req, &d);
3947 err = fuse_fs_removexattr(f->fs, path, name);
3948 fuse_finish_interrupt(f, req, &d);
3949 free_path(f, ino, path);
3950 }
3951 reply_err(req, err);
3952 }
3953
locks_conflict(struct node * node,const struct lock * lock)3954 static struct lock *locks_conflict(struct node *node, const struct lock *lock)
3955 {
3956 struct lock *l;
3957
3958 for (l = node->locks; l; l = l->next)
3959 if (l->owner != lock->owner &&
3960 lock->start <= l->end && l->start <= lock->end &&
3961 (l->type == F_WRLCK || lock->type == F_WRLCK))
3962 break;
3963
3964 return l;
3965 }
3966
delete_lock(struct lock ** lockp)3967 static void delete_lock(struct lock **lockp)
3968 {
3969 struct lock *l = *lockp;
3970 *lockp = l->next;
3971 free(l);
3972 }
3973
insert_lock(struct lock ** pos,struct lock * lock)3974 static void insert_lock(struct lock **pos, struct lock *lock)
3975 {
3976 lock->next = *pos;
3977 *pos = lock;
3978 }
3979
locks_insert(struct node * node,struct lock * lock)3980 static int locks_insert(struct node *node, struct lock *lock)
3981 {
3982 struct lock **lp;
3983 struct lock *newl1 = NULL;
3984 struct lock *newl2 = NULL;
3985
3986 if (lock->type != F_UNLCK || lock->start != 0 ||
3987 lock->end != OFFSET_MAX) {
3988 newl1 = malloc(sizeof(struct lock));
3989 newl2 = malloc(sizeof(struct lock));
3990
3991 if (!newl1 || !newl2) {
3992 free(newl1);
3993 free(newl2);
3994 return -ENOLCK;
3995 }
3996 }
3997
3998 for (lp = &node->locks; *lp;) {
3999 struct lock *l = *lp;
4000 if (l->owner != lock->owner)
4001 goto skip;
4002
4003 if (lock->type == l->type) {
4004 if (l->end < lock->start - 1)
4005 goto skip;
4006 if (lock->end < l->start - 1)
4007 break;
4008 if (l->start <= lock->start && lock->end <= l->end)
4009 goto out;
4010 if (l->start < lock->start)
4011 lock->start = l->start;
4012 if (lock->end < l->end)
4013 lock->end = l->end;
4014 goto delete;
4015 } else {
4016 if (l->end < lock->start)
4017 goto skip;
4018 if (lock->end < l->start)
4019 break;
4020 if (lock->start <= l->start && l->end <= lock->end)
4021 goto delete;
4022 if (l->end <= lock->end) {
4023 l->end = lock->start - 1;
4024 goto skip;
4025 }
4026 if (lock->start <= l->start) {
4027 l->start = lock->end + 1;
4028 break;
4029 }
4030 *newl2 = *l;
4031 newl2->start = lock->end + 1;
4032 l->end = lock->start - 1;
4033 insert_lock(&l->next, newl2);
4034 newl2 = NULL;
4035 }
4036 skip:
4037 lp = &l->next;
4038 continue;
4039
4040 delete:
4041 delete_lock(lp);
4042 }
4043 if (lock->type != F_UNLCK) {
4044 *newl1 = *lock;
4045 insert_lock(lp, newl1);
4046 newl1 = NULL;
4047 }
4048 out:
4049 free(newl1);
4050 free(newl2);
4051 return 0;
4052 }
4053
flock_to_lock(struct flock * flock,struct lock * lock)4054 static void flock_to_lock(struct flock *flock, struct lock *lock)
4055 {
4056 memset(lock, 0, sizeof(struct lock));
4057 lock->type = flock->l_type;
4058 lock->start = flock->l_start;
4059 lock->end =
4060 flock->l_len ? flock->l_start + flock->l_len - 1 : OFFSET_MAX;
4061 lock->pid = flock->l_pid;
4062 }
4063
lock_to_flock(struct lock * lock,struct flock * flock)4064 static void lock_to_flock(struct lock *lock, struct flock *flock)
4065 {
4066 flock->l_type = lock->type;
4067 flock->l_start = lock->start;
4068 flock->l_len =
4069 (lock->end == OFFSET_MAX) ? 0 : lock->end - lock->start + 1;
4070 flock->l_pid = lock->pid;
4071 }
4072
fuse_flush_common(struct fuse * f,fuse_req_t req,fuse_ino_t ino,const char * path,struct fuse_file_info * fi)4073 static int fuse_flush_common(struct fuse *f, fuse_req_t req, fuse_ino_t ino,
4074 const char *path, struct fuse_file_info *fi)
4075 {
4076 struct fuse_intr_data d;
4077 struct flock lock;
4078 struct lock l;
4079 int err;
4080 int errlock;
4081
4082 fuse_prepare_interrupt(f, req, &d);
4083 memset(&lock, 0, sizeof(lock));
4084 lock.l_type = F_UNLCK;
4085 lock.l_whence = SEEK_SET;
4086 err = fuse_fs_flush(f->fs, path, fi);
4087 errlock = fuse_fs_lock(f->fs, path, fi, F_SETLK, &lock);
4088 fuse_finish_interrupt(f, req, &d);
4089
4090 if (errlock != -ENOSYS) {
4091 flock_to_lock(&lock, &l);
4092 l.owner = fi->lock_owner;
4093 pthread_mutex_lock(&f->lock);
4094 locks_insert(get_node(f, ino), &l);
4095 pthread_mutex_unlock(&f->lock);
4096
4097 /* if op.lock() is defined FLUSH is needed regardless
4098 of op.flush() */
4099 if (err == -ENOSYS)
4100 err = 0;
4101 }
4102 return err;
4103 }
4104
fuse_lib_release(fuse_req_t req,fuse_ino_t ino,struct fuse_file_info * fi)4105 static void fuse_lib_release(fuse_req_t req, fuse_ino_t ino,
4106 struct fuse_file_info *fi)
4107 {
4108 struct fuse *f = req_fuse_prepare(req);
4109 struct fuse_intr_data d;
4110 char *path;
4111 int err = 0;
4112
4113 get_path_nullok(f, ino, &path);
4114 if (fi->flush) {
4115 err = fuse_flush_common(f, req, ino, path, fi);
4116 if (err == -ENOSYS)
4117 err = 0;
4118 }
4119
4120 fuse_prepare_interrupt(f, req, &d);
4121 fuse_do_release(f, ino, path, fi);
4122 fuse_finish_interrupt(f, req, &d);
4123 free_path(f, ino, path);
4124
4125 reply_err(req, err);
4126 }
4127
fuse_lib_flush(fuse_req_t req,fuse_ino_t ino,struct fuse_file_info * fi)4128 static void fuse_lib_flush(fuse_req_t req, fuse_ino_t ino,
4129 struct fuse_file_info *fi)
4130 {
4131 struct fuse *f = req_fuse_prepare(req);
4132 char *path;
4133 int err;
4134
4135 get_path_nullok(f, ino, &path);
4136 err = fuse_flush_common(f, req, ino, path, fi);
4137 free_path(f, ino, path);
4138
4139 reply_err(req, err);
4140 }
4141
fuse_lock_common(fuse_req_t req,fuse_ino_t ino,struct fuse_file_info * fi,struct flock * lock,int cmd)4142 static int fuse_lock_common(fuse_req_t req, fuse_ino_t ino,
4143 struct fuse_file_info *fi, struct flock *lock,
4144 int cmd)
4145 {
4146 struct fuse *f = req_fuse_prepare(req);
4147 char *path;
4148 int err;
4149
4150 err = get_path_nullok(f, ino, &path);
4151 if (!err) {
4152 struct fuse_intr_data d;
4153 fuse_prepare_interrupt(f, req, &d);
4154 err = fuse_fs_lock(f->fs, path, fi, cmd, lock);
4155 fuse_finish_interrupt(f, req, &d);
4156 free_path(f, ino, path);
4157 }
4158 return err;
4159 }
4160
fuse_lib_getlk(fuse_req_t req,fuse_ino_t ino,struct fuse_file_info * fi,struct flock * lock)4161 static void fuse_lib_getlk(fuse_req_t req, fuse_ino_t ino,
4162 struct fuse_file_info *fi, struct flock *lock)
4163 {
4164 int err;
4165 struct lock l;
4166 struct lock *conflict;
4167 struct fuse *f = req_fuse(req);
4168
4169 flock_to_lock(lock, &l);
4170 l.owner = fi->lock_owner;
4171 pthread_mutex_lock(&f->lock);
4172 conflict = locks_conflict(get_node(f, ino), &l);
4173 if (conflict)
4174 lock_to_flock(conflict, lock);
4175 pthread_mutex_unlock(&f->lock);
4176 if (!conflict)
4177 err = fuse_lock_common(req, ino, fi, lock, F_GETLK);
4178 else
4179 err = 0;
4180
4181 if (!err)
4182 fuse_reply_lock(req, lock);
4183 else
4184 reply_err(req, err);
4185 }
4186
fuse_lib_setlk(fuse_req_t req,fuse_ino_t ino,struct fuse_file_info * fi,struct flock * lock,int sleep)4187 static void fuse_lib_setlk(fuse_req_t req, fuse_ino_t ino,
4188 struct fuse_file_info *fi, struct flock *lock,
4189 int sleep)
4190 {
4191 int err = fuse_lock_common(req, ino, fi, lock,
4192 sleep ? F_SETLKW : F_SETLK);
4193 if (!err) {
4194 struct fuse *f = req_fuse(req);
4195 struct lock l;
4196 flock_to_lock(lock, &l);
4197 l.owner = fi->lock_owner;
4198 pthread_mutex_lock(&f->lock);
4199 locks_insert(get_node(f, ino), &l);
4200 pthread_mutex_unlock(&f->lock);
4201 }
4202 reply_err(req, err);
4203 }
4204
fuse_lib_flock(fuse_req_t req,fuse_ino_t ino,struct fuse_file_info * fi,int op)4205 static void fuse_lib_flock(fuse_req_t req, fuse_ino_t ino,
4206 struct fuse_file_info *fi, int op)
4207 {
4208 struct fuse *f = req_fuse_prepare(req);
4209 char *path;
4210 int err;
4211
4212 err = get_path_nullok(f, ino, &path);
4213 if (err == 0) {
4214 struct fuse_intr_data d;
4215 fuse_prepare_interrupt(f, req, &d);
4216 err = fuse_fs_flock(f->fs, path, fi, op);
4217 fuse_finish_interrupt(f, req, &d);
4218 free_path(f, ino, path);
4219 }
4220 reply_err(req, err);
4221 }
4222
fuse_lib_bmap(fuse_req_t req,fuse_ino_t ino,size_t blocksize,uint64_t idx)4223 static void fuse_lib_bmap(fuse_req_t req, fuse_ino_t ino, size_t blocksize,
4224 uint64_t idx)
4225 {
4226 struct fuse *f = req_fuse_prepare(req);
4227 struct fuse_intr_data d;
4228 char *path;
4229 int err;
4230
4231 err = get_path(f, ino, &path);
4232 if (!err) {
4233 fuse_prepare_interrupt(f, req, &d);
4234 err = fuse_fs_bmap(f->fs, path, blocksize, &idx);
4235 fuse_finish_interrupt(f, req, &d);
4236 free_path(f, ino, path);
4237 }
4238 if (!err)
4239 fuse_reply_bmap(req, idx);
4240 else
4241 reply_err(req, err);
4242 }
4243
fuse_lib_ioctl(fuse_req_t req,fuse_ino_t ino,unsigned int cmd,void * arg,struct fuse_file_info * llfi,unsigned int flags,const void * in_buf,size_t in_bufsz,size_t out_bufsz)4244 static void fuse_lib_ioctl(fuse_req_t req, fuse_ino_t ino, unsigned int cmd,
4245 void *arg, struct fuse_file_info *llfi,
4246 unsigned int flags, const void *in_buf,
4247 size_t in_bufsz, size_t out_bufsz)
4248 {
4249 struct fuse *f = req_fuse_prepare(req);
4250 struct fuse_intr_data d;
4251 struct fuse_file_info fi;
4252 char *path, *out_buf = NULL;
4253 int err;
4254
4255 err = -EPERM;
4256 if (flags & FUSE_IOCTL_UNRESTRICTED)
4257 goto err;
4258
4259 if (flags & FUSE_IOCTL_DIR)
4260 get_dirhandle(llfi, &fi);
4261 else
4262 fi = *llfi;
4263
4264 if (out_bufsz) {
4265 err = -ENOMEM;
4266 out_buf = malloc(out_bufsz);
4267 if (!out_buf)
4268 goto err;
4269 }
4270
4271 assert(!in_bufsz || !out_bufsz || in_bufsz == out_bufsz);
4272 if (out_buf && in_bufsz)
4273 memcpy(out_buf, in_buf, in_bufsz);
4274
4275 err = get_path_nullok(f, ino, &path);
4276 if (err)
4277 goto err;
4278
4279 fuse_prepare_interrupt(f, req, &d);
4280
4281 err = fuse_fs_ioctl(f->fs, path, cmd, arg, &fi, flags,
4282 out_buf ? out_buf : (void *)in_buf);
4283
4284 fuse_finish_interrupt(f, req, &d);
4285 free_path(f, ino, path);
4286
4287 fuse_reply_ioctl(req, err, out_buf, out_bufsz);
4288 goto out;
4289 err:
4290 reply_err(req, err);
4291 out:
4292 free(out_buf);
4293 }
4294
fuse_lib_poll(fuse_req_t req,fuse_ino_t ino,struct fuse_file_info * fi,struct fuse_pollhandle * ph)4295 static void fuse_lib_poll(fuse_req_t req, fuse_ino_t ino,
4296 struct fuse_file_info *fi, struct fuse_pollhandle *ph)
4297 {
4298 struct fuse *f = req_fuse_prepare(req);
4299 struct fuse_intr_data d;
4300 char *path;
4301 int err;
4302 unsigned revents = 0;
4303
4304 err = get_path_nullok(f, ino, &path);
4305 if (!err) {
4306 fuse_prepare_interrupt(f, req, &d);
4307 err = fuse_fs_poll(f->fs, path, fi, ph, &revents);
4308 fuse_finish_interrupt(f, req, &d);
4309 free_path(f, ino, path);
4310 }
4311 if (!err)
4312 fuse_reply_poll(req, revents);
4313 else
4314 reply_err(req, err);
4315 }
4316
fuse_lib_fallocate(fuse_req_t req,fuse_ino_t ino,int mode,off_t offset,off_t length,struct fuse_file_info * fi)4317 static void fuse_lib_fallocate(fuse_req_t req, fuse_ino_t ino, int mode,
4318 off_t offset, off_t length, struct fuse_file_info *fi)
4319 {
4320 struct fuse *f = req_fuse_prepare(req);
4321 struct fuse_intr_data d;
4322 char *path;
4323 int err;
4324
4325 err = get_path_nullok(f, ino, &path);
4326 if (!err) {
4327 fuse_prepare_interrupt(f, req, &d);
4328 err = fuse_fs_fallocate(f->fs, path, mode, offset, length, fi);
4329 fuse_finish_interrupt(f, req, &d);
4330 free_path(f, ino, path);
4331 }
4332 reply_err(req, err);
4333 }
4334
fuse_lib_copy_file_range(fuse_req_t req,fuse_ino_t nodeid_in,off_t off_in,struct fuse_file_info * fi_in,fuse_ino_t nodeid_out,off_t off_out,struct fuse_file_info * fi_out,size_t len,int flags)4335 static void fuse_lib_copy_file_range(fuse_req_t req, fuse_ino_t nodeid_in,
4336 off_t off_in, struct fuse_file_info *fi_in,
4337 fuse_ino_t nodeid_out, off_t off_out,
4338 struct fuse_file_info *fi_out, size_t len,
4339 int flags)
4340 {
4341 struct fuse *f = req_fuse_prepare(req);
4342 struct fuse_intr_data d;
4343 char *path_in, *path_out;
4344 int err;
4345 ssize_t res;
4346
4347 err = get_path_nullok(f, nodeid_in, &path_in);
4348 if (err) {
4349 reply_err(req, err);
4350 return;
4351 }
4352
4353 err = get_path_nullok(f, nodeid_out, &path_out);
4354 if (err) {
4355 free_path(f, nodeid_in, path_in);
4356 reply_err(req, err);
4357 return;
4358 }
4359
4360 fuse_prepare_interrupt(f, req, &d);
4361 res = fuse_fs_copy_file_range(f->fs, path_in, fi_in, off_in, path_out,
4362 fi_out, off_out, len, flags);
4363 fuse_finish_interrupt(f, req, &d);
4364
4365 if (res >= 0)
4366 fuse_reply_write(req, res);
4367 else
4368 reply_err(req, res);
4369
4370 free_path(f, nodeid_in, path_in);
4371 free_path(f, nodeid_out, path_out);
4372 }
4373
fuse_lib_lseek(fuse_req_t req,fuse_ino_t ino,off_t off,int whence,struct fuse_file_info * fi)4374 static void fuse_lib_lseek(fuse_req_t req, fuse_ino_t ino, off_t off, int whence,
4375 struct fuse_file_info *fi)
4376 {
4377 struct fuse *f = req_fuse_prepare(req);
4378 struct fuse_intr_data d;
4379 char *path;
4380 int err;
4381 off_t res;
4382
4383 err = get_path(f, ino, &path);
4384 if (err) {
4385 reply_err(req, err);
4386 return;
4387 }
4388
4389 fuse_prepare_interrupt(f, req, &d);
4390 res = fuse_fs_lseek(f->fs, path, off, whence, fi);
4391 fuse_finish_interrupt(f, req, &d);
4392 free_path(f, ino, path);
4393 if (res >= 0)
4394 fuse_reply_lseek(req, res);
4395 else
4396 reply_err(req, res);
4397 }
4398
clean_delay(struct fuse * f)4399 static int clean_delay(struct fuse *f)
4400 {
4401 /*
4402 * This is calculating the delay between clean runs. To
4403 * reduce the number of cleans we are doing them 10 times
4404 * within the remember window.
4405 */
4406 int min_sleep = 60;
4407 int max_sleep = 3600;
4408 int sleep_time = f->conf.remember / 10;
4409
4410 if (sleep_time > max_sleep)
4411 return max_sleep;
4412 if (sleep_time < min_sleep)
4413 return min_sleep;
4414 return sleep_time;
4415 }
4416
fuse_clean_cache(struct fuse * f)4417 int fuse_clean_cache(struct fuse *f)
4418 {
4419 struct node_lru *lnode;
4420 struct list_head *curr, *next;
4421 struct node *node;
4422 struct timespec now;
4423
4424 pthread_mutex_lock(&f->lock);
4425
4426 curr_time(&now);
4427
4428 for (curr = f->lru_table.next; curr != &f->lru_table; curr = next) {
4429 double age;
4430
4431 next = curr->next;
4432 lnode = list_entry(curr, struct node_lru, lru);
4433 node = &lnode->node;
4434
4435 age = diff_timespec(&now, &lnode->forget_time);
4436 if (age <= f->conf.remember)
4437 break;
4438
4439 assert(node->nlookup == 1);
4440
4441 /* Don't forget active directories */
4442 if (node->refctr > 1)
4443 continue;
4444
4445 node->nlookup = 0;
4446 unhash_name(f, node);
4447 unref_node(f, node);
4448 }
4449 pthread_mutex_unlock(&f->lock);
4450
4451 return clean_delay(f);
4452 }
4453
4454 static struct fuse_lowlevel_ops fuse_path_ops = {
4455 .init = fuse_lib_init,
4456 .destroy = fuse_lib_destroy,
4457 .lookup = fuse_lib_lookup,
4458 .forget = fuse_lib_forget,
4459 .forget_multi = fuse_lib_forget_multi,
4460 .getattr = fuse_lib_getattr,
4461 .setattr = fuse_lib_setattr,
4462 .access = fuse_lib_access,
4463 .readlink = fuse_lib_readlink,
4464 .mknod = fuse_lib_mknod,
4465 .mkdir = fuse_lib_mkdir,
4466 .unlink = fuse_lib_unlink,
4467 .rmdir = fuse_lib_rmdir,
4468 .symlink = fuse_lib_symlink,
4469 .rename = fuse_lib_rename,
4470 .link = fuse_lib_link,
4471 .create = fuse_lib_create,
4472 .open = fuse_lib_open,
4473 .read = fuse_lib_read,
4474 .write_buf = fuse_lib_write_buf,
4475 .flush = fuse_lib_flush,
4476 .release = fuse_lib_release,
4477 .fsync = fuse_lib_fsync,
4478 .opendir = fuse_lib_opendir,
4479 .readdir = fuse_lib_readdir,
4480 .readdirplus = fuse_lib_readdirplus,
4481 .releasedir = fuse_lib_releasedir,
4482 .fsyncdir = fuse_lib_fsyncdir,
4483 .statfs = fuse_lib_statfs,
4484 .setxattr = fuse_lib_setxattr,
4485 .getxattr = fuse_lib_getxattr,
4486 .listxattr = fuse_lib_listxattr,
4487 .removexattr = fuse_lib_removexattr,
4488 .getlk = fuse_lib_getlk,
4489 .setlk = fuse_lib_setlk,
4490 .flock = fuse_lib_flock,
4491 .bmap = fuse_lib_bmap,
4492 .ioctl = fuse_lib_ioctl,
4493 .poll = fuse_lib_poll,
4494 .fallocate = fuse_lib_fallocate,
4495 .copy_file_range = fuse_lib_copy_file_range,
4496 .lseek = fuse_lib_lseek,
4497 };
4498
fuse_notify_poll(struct fuse_pollhandle * ph)4499 int fuse_notify_poll(struct fuse_pollhandle *ph)
4500 {
4501 return fuse_lowlevel_notify_poll(ph);
4502 }
4503
fuse_get_session(struct fuse * f)4504 struct fuse_session *fuse_get_session(struct fuse *f)
4505 {
4506 return f->se;
4507 }
4508
fuse_session_loop_remember(struct fuse * f)4509 static int fuse_session_loop_remember(struct fuse *f)
4510 {
4511 struct fuse_session *se = f->se;
4512 int res = 0;
4513 struct timespec now;
4514 time_t next_clean;
4515 struct pollfd fds = {
4516 .fd = se->fd,
4517 .events = POLLIN
4518 };
4519 struct fuse_buf fbuf = {
4520 .mem = NULL,
4521 };
4522
4523 curr_time(&now);
4524 next_clean = now.tv_sec;
4525 while (!fuse_session_exited(se)) {
4526 unsigned timeout;
4527
4528 curr_time(&now);
4529 if (now.tv_sec < next_clean)
4530 timeout = next_clean - now.tv_sec;
4531 else
4532 timeout = 0;
4533
4534 res = poll(&fds, 1, timeout * 1000);
4535 if (res == -1) {
4536 if (errno == -EINTR)
4537 continue;
4538 else
4539 break;
4540 } else if (res > 0) {
4541 res = fuse_session_receive_buf_int(se, &fbuf, NULL);
4542
4543 if (res == -EINTR)
4544 continue;
4545 if (res <= 0)
4546 break;
4547
4548 fuse_session_process_buf_int(se, &fbuf, NULL);
4549 } else {
4550 timeout = fuse_clean_cache(f);
4551 curr_time(&now);
4552 next_clean = now.tv_sec + timeout;
4553 }
4554 }
4555
4556 free(fbuf.mem);
4557 fuse_session_reset(se);
4558 return res < 0 ? -1 : 0;
4559 }
4560
fuse_loop(struct fuse * f)4561 int fuse_loop(struct fuse *f)
4562 {
4563 if (!f)
4564 return -1;
4565
4566 if (lru_enabled(f))
4567 return fuse_session_loop_remember(f);
4568
4569 return fuse_session_loop(f->se);
4570 }
4571
4572 FUSE_SYMVER(".symver fuse_loop_mt_32,fuse_loop_mt@@FUSE_3.2");
fuse_loop_mt_32(struct fuse * f,struct fuse_loop_config * config)4573 int fuse_loop_mt_32(struct fuse *f, struct fuse_loop_config *config)
4574 {
4575 if (f == NULL)
4576 return -1;
4577
4578 int res = fuse_start_cleanup_thread(f);
4579 if (res)
4580 return -1;
4581
4582 res = fuse_session_loop_mt_32(fuse_get_session(f), config);
4583 fuse_stop_cleanup_thread(f);
4584 return res;
4585 }
4586
4587 int fuse_loop_mt_31(struct fuse *f, int clone_fd);
4588 FUSE_SYMVER(".symver fuse_loop_mt_31,fuse_loop_mt@FUSE_3.0");
fuse_loop_mt_31(struct fuse * f,int clone_fd)4589 int fuse_loop_mt_31(struct fuse *f, int clone_fd)
4590 {
4591 struct fuse_loop_config config;
4592 config.clone_fd = clone_fd;
4593 config.max_idle_threads = 10;
4594 return fuse_loop_mt_32(f, &config);
4595 }
4596
fuse_exit(struct fuse * f)4597 void fuse_exit(struct fuse *f)
4598 {
4599 fuse_session_exit(f->se);
4600 }
4601
fuse_get_context(void)4602 struct fuse_context *fuse_get_context(void)
4603 {
4604 struct fuse_context_i *c = fuse_get_context_internal();
4605
4606 if (c)
4607 return &c->ctx;
4608 else
4609 return NULL;
4610 }
4611
fuse_getgroups(int size,gid_t list[])4612 int fuse_getgroups(int size, gid_t list[])
4613 {
4614 struct fuse_context_i *c = fuse_get_context_internal();
4615 if (!c)
4616 return -EINVAL;
4617
4618 return fuse_req_getgroups(c->req, size, list);
4619 }
4620
fuse_interrupted(void)4621 int fuse_interrupted(void)
4622 {
4623 struct fuse_context_i *c = fuse_get_context_internal();
4624
4625 if (c)
4626 return fuse_req_interrupted(c->req);
4627 else
4628 return 0;
4629 }
4630
fuse_invalidate_path(struct fuse * f,const char * path)4631 int fuse_invalidate_path(struct fuse *f, const char *path) {
4632 fuse_ino_t ino;
4633 int err = lookup_path_in_cache(f, path, &ino);
4634 if (err) {
4635 return err;
4636 }
4637
4638 return fuse_lowlevel_notify_inval_inode(f->se, ino, 0, 0);
4639 }
4640
4641 #define FUSE_LIB_OPT(t, p, v) { t, offsetof(struct fuse_config, p), v }
4642
4643 static const struct fuse_opt fuse_lib_opts[] = {
4644 FUSE_OPT_KEY("debug", FUSE_OPT_KEY_KEEP),
4645 FUSE_OPT_KEY("-d", FUSE_OPT_KEY_KEEP),
4646 FUSE_LIB_OPT("debug", debug, 1),
4647 FUSE_LIB_OPT("-d", debug, 1),
4648 FUSE_LIB_OPT("kernel_cache", kernel_cache, 1),
4649 FUSE_LIB_OPT("auto_cache", auto_cache, 1),
4650 FUSE_LIB_OPT("noauto_cache", auto_cache, 0),
4651 FUSE_LIB_OPT("umask=", set_mode, 1),
4652 FUSE_LIB_OPT("umask=%o", umask, 0),
4653 FUSE_LIB_OPT("uid=", set_uid, 1),
4654 FUSE_LIB_OPT("uid=%d", uid, 0),
4655 FUSE_LIB_OPT("gid=", set_gid, 1),
4656 FUSE_LIB_OPT("gid=%d", gid, 0),
4657 FUSE_LIB_OPT("entry_timeout=%lf", entry_timeout, 0),
4658 FUSE_LIB_OPT("attr_timeout=%lf", attr_timeout, 0),
4659 FUSE_LIB_OPT("ac_attr_timeout=%lf", ac_attr_timeout, 0),
4660 FUSE_LIB_OPT("ac_attr_timeout=", ac_attr_timeout_set, 1),
4661 FUSE_LIB_OPT("negative_timeout=%lf", negative_timeout, 0),
4662 FUSE_LIB_OPT("noforget", remember, -1),
4663 FUSE_LIB_OPT("remember=%u", remember, 0),
4664 FUSE_LIB_OPT("modules=%s", modules, 0),
4665 FUSE_OPT_END
4666 };
4667
fuse_lib_opt_proc(void * data,const char * arg,int key,struct fuse_args * outargs)4668 static int fuse_lib_opt_proc(void *data, const char *arg, int key,
4669 struct fuse_args *outargs)
4670 {
4671 (void) arg; (void) outargs; (void) data; (void) key;
4672
4673 /* Pass through unknown options */
4674 return 1;
4675 }
4676
4677
4678 static const struct fuse_opt fuse_help_opts[] = {
4679 FUSE_LIB_OPT("modules=%s", modules, 1),
4680 FUSE_OPT_KEY("modules=%s", FUSE_OPT_KEY_KEEP),
4681 FUSE_OPT_END
4682 };
4683
print_module_help(const char * name,fuse_module_factory_t * fac)4684 static void print_module_help(const char *name,
4685 fuse_module_factory_t *fac)
4686 {
4687 struct fuse_args a = FUSE_ARGS_INIT(0, NULL);
4688 if (fuse_opt_add_arg(&a, "") == -1 ||
4689 fuse_opt_add_arg(&a, "-h") == -1)
4690 return;
4691 printf("\nOptions for %s module:\n", name);
4692 (*fac)(&a, NULL);
4693 fuse_opt_free_args(&a);
4694 }
4695
fuse_lib_help(struct fuse_args * args)4696 void fuse_lib_help(struct fuse_args *args)
4697 {
4698 /* These are not all options, but only the ones that
4699 may be of interest to an end-user */
4700 printf(
4701 " -o kernel_cache cache files in kernel\n"
4702 " -o [no]auto_cache enable caching based on modification times (off)\n"
4703 " -o umask=M set file permissions (octal)\n"
4704 " -o uid=N set file owner\n"
4705 " -o gid=N set file group\n"
4706 " -o entry_timeout=T cache timeout for names (1.0s)\n"
4707 " -o negative_timeout=T cache timeout for deleted names (0.0s)\n"
4708 " -o attr_timeout=T cache timeout for attributes (1.0s)\n"
4709 " -o ac_attr_timeout=T auto cache timeout for attributes (attr_timeout)\n"
4710 " -o noforget never forget cached inodes\n"
4711 " -o remember=T remember cached inodes for T seconds (0s)\n"
4712 " -o modules=M1[:M2...] names of modules to push onto filesystem stack\n");
4713
4714
4715 /* Print low-level help */
4716 fuse_lowlevel_help();
4717
4718 /* Print help for builtin modules */
4719 print_module_help("subdir", &fuse_module_subdir_factory);
4720 #ifdef HAVE_ICONV
4721 print_module_help("iconv", &fuse_module_iconv_factory);
4722 #endif
4723
4724 /* Parse command line options in case we need to
4725 activate more modules */
4726 struct fuse_config conf = { .modules = NULL };
4727 if (fuse_opt_parse(args, &conf, fuse_help_opts,
4728 fuse_lib_opt_proc) == -1
4729 || !conf.modules)
4730 return;
4731
4732 char *module;
4733 char *next;
4734 struct fuse_module *m;
4735
4736 // Iterate over all modules
4737 for (module = conf.modules; module; module = next) {
4738 char *p;
4739 for (p = module; *p && *p != ':'; p++);
4740 next = *p ? p + 1 : NULL;
4741 *p = '\0';
4742
4743 m = fuse_get_module(module);
4744 if (m)
4745 print_module_help(module, &m->factory);
4746 }
4747 }
4748
4749
4750
fuse_init_intr_signal(int signum,int * installed)4751 static int fuse_init_intr_signal(int signum, int *installed)
4752 {
4753 struct sigaction old_sa;
4754
4755 if (sigaction(signum, NULL, &old_sa) == -1) {
4756 perror("fuse: cannot get old signal handler");
4757 return -1;
4758 }
4759
4760 if (old_sa.sa_handler == SIG_DFL) {
4761 struct sigaction sa;
4762
4763 memset(&sa, 0, sizeof(struct sigaction));
4764 sa.sa_handler = fuse_intr_sighandler;
4765 sigemptyset(&sa.sa_mask);
4766
4767 if (sigaction(signum, &sa, NULL) == -1) {
4768 perror("fuse: cannot set interrupt signal handler");
4769 return -1;
4770 }
4771 *installed = 1;
4772 }
4773 return 0;
4774 }
4775
fuse_restore_intr_signal(int signum)4776 static void fuse_restore_intr_signal(int signum)
4777 {
4778 struct sigaction sa;
4779
4780 memset(&sa, 0, sizeof(struct sigaction));
4781 sa.sa_handler = SIG_DFL;
4782 sigaction(signum, &sa, NULL);
4783 }
4784
4785
fuse_push_module(struct fuse * f,const char * module,struct fuse_args * args)4786 static int fuse_push_module(struct fuse *f, const char *module,
4787 struct fuse_args *args)
4788 {
4789 struct fuse_fs *fs[2] = { f->fs, NULL };
4790 struct fuse_fs *newfs;
4791 struct fuse_module *m = fuse_get_module(module);
4792
4793 if (!m)
4794 return -1;
4795
4796 newfs = m->factory(args, fs);
4797 if (!newfs) {
4798 fuse_put_module(m);
4799 return -1;
4800 }
4801 newfs->m = m;
4802 f->fs = newfs;
4803 return 0;
4804 }
4805
fuse_fs_new(const struct fuse_operations * op,size_t op_size,void * user_data)4806 struct fuse_fs *fuse_fs_new(const struct fuse_operations *op, size_t op_size,
4807 void *user_data)
4808 {
4809 struct fuse_fs *fs;
4810
4811 if (sizeof(struct fuse_operations) < op_size) {
4812 fuse_log(FUSE_LOG_ERR, "fuse: warning: library too old, some operations may not not work\n");
4813 op_size = sizeof(struct fuse_operations);
4814 }
4815
4816 fs = (struct fuse_fs *) calloc(1, sizeof(struct fuse_fs));
4817 if (!fs) {
4818 fuse_log(FUSE_LOG_ERR, "fuse: failed to allocate fuse_fs object\n");
4819 return NULL;
4820 }
4821
4822 fs->user_data = user_data;
4823 if (op)
4824 memcpy(&fs->op, op, op_size);
4825 return fs;
4826 }
4827
node_table_init(struct node_table * t)4828 static int node_table_init(struct node_table *t)
4829 {
4830 t->size = NODE_TABLE_MIN_SIZE;
4831 t->array = (struct node **) calloc(1, sizeof(struct node *) * t->size);
4832 if (t->array == NULL) {
4833 fuse_log(FUSE_LOG_ERR, "fuse: memory allocation failed\n");
4834 return -1;
4835 }
4836 t->use = 0;
4837 t->split = 0;
4838
4839 return 0;
4840 }
4841
fuse_prune_nodes(void * fuse)4842 static void *fuse_prune_nodes(void *fuse)
4843 {
4844 struct fuse *f = fuse;
4845 int sleep_time;
4846
4847 while(1) {
4848 sleep_time = fuse_clean_cache(f);
4849 sleep(sleep_time);
4850 }
4851 return NULL;
4852 }
4853
fuse_start_cleanup_thread(struct fuse * f)4854 int fuse_start_cleanup_thread(struct fuse *f)
4855 {
4856 if (lru_enabled(f))
4857 return fuse_start_thread(&f->prune_thread, fuse_prune_nodes, f);
4858
4859 return 0;
4860 }
4861
fuse_stop_cleanup_thread(struct fuse * f)4862 void fuse_stop_cleanup_thread(struct fuse *f)
4863 {
4864 if (lru_enabled(f)) {
4865 pthread_mutex_lock(&f->lock);
4866 pthread_cancel(f->prune_thread);
4867 pthread_mutex_unlock(&f->lock);
4868 pthread_join(f->prune_thread, NULL);
4869 }
4870 }
4871
4872
4873 FUSE_SYMVER(".symver fuse_new_31,fuse_new@@FUSE_3.1");
fuse_new_31(struct fuse_args * args,const struct fuse_operations * op,size_t op_size,void * user_data)4874 struct fuse *fuse_new_31(struct fuse_args *args,
4875 const struct fuse_operations *op,
4876 size_t op_size, void *user_data)
4877 {
4878 struct fuse *f;
4879 struct node *root;
4880 struct fuse_fs *fs;
4881 struct fuse_lowlevel_ops llop = fuse_path_ops;
4882
4883 f = (struct fuse *) calloc(1, sizeof(struct fuse));
4884 if (f == NULL) {
4885 fuse_log(FUSE_LOG_ERR, "fuse: failed to allocate fuse object\n");
4886 goto out;
4887 }
4888
4889 f->conf.entry_timeout = 1.0;
4890 f->conf.attr_timeout = 1.0;
4891 f->conf.negative_timeout = 0.0;
4892 f->conf.intr_signal = FUSE_DEFAULT_INTR_SIGNAL;
4893
4894 /* Parse options */
4895 if (fuse_opt_parse(args, &f->conf, fuse_lib_opts,
4896 fuse_lib_opt_proc) == -1)
4897 goto out_free;
4898
4899 pthread_mutex_lock(&fuse_context_lock);
4900 static int builtin_modules_registered = 0;
4901 /* Have the builtin modules already been registered? */
4902 if (builtin_modules_registered == 0) {
4903 /* If not, register them. */
4904 fuse_register_module("subdir", fuse_module_subdir_factory, NULL);
4905 #ifdef HAVE_ICONV
4906 fuse_register_module("iconv", fuse_module_iconv_factory, NULL);
4907 #endif
4908 builtin_modules_registered= 1;
4909 }
4910 pthread_mutex_unlock(&fuse_context_lock);
4911
4912 if (fuse_create_context_key() == -1)
4913 goto out_free;
4914
4915 fs = fuse_fs_new(op, op_size, user_data);
4916 if (!fs)
4917 goto out_delete_context_key;
4918
4919 f->fs = fs;
4920
4921 /* Oh f**k, this is ugly! */
4922 if (!fs->op.lock) {
4923 llop.getlk = NULL;
4924 llop.setlk = NULL;
4925 }
4926
4927 f->pagesize = getpagesize();
4928 init_list_head(&f->partial_slabs);
4929 init_list_head(&f->full_slabs);
4930 init_list_head(&f->lru_table);
4931
4932 if (f->conf.modules) {
4933 char *module;
4934 char *next;
4935
4936 for (module = f->conf.modules; module; module = next) {
4937 char *p;
4938 for (p = module; *p && *p != ':'; p++);
4939 next = *p ? p + 1 : NULL;
4940 *p = '\0';
4941 if (module[0] &&
4942 fuse_push_module(f, module, args) == -1)
4943 goto out_free_fs;
4944 }
4945 }
4946
4947 if (!f->conf.ac_attr_timeout_set)
4948 f->conf.ac_attr_timeout = f->conf.attr_timeout;
4949
4950 #if defined(__FreeBSD__) || defined(__NetBSD__)
4951 /*
4952 * In FreeBSD, we always use these settings as inode numbers
4953 * are needed to make getcwd(3) work.
4954 */
4955 f->conf.readdir_ino = 1;
4956 #endif
4957
4958 f->se = fuse_session_new(args, &llop, sizeof(llop), f);
4959 if (f->se == NULL)
4960 goto out_free_fs;
4961
4962 if (f->conf.debug) {
4963 fuse_log(FUSE_LOG_DEBUG, "nullpath_ok: %i\n", f->conf.nullpath_ok);
4964 }
4965
4966 /* Trace topmost layer by default */
4967 f->fs->debug = f->conf.debug;
4968 f->ctr = 0;
4969 f->generation = 0;
4970 if (node_table_init(&f->name_table) == -1)
4971 goto out_free_session;
4972
4973 if (node_table_init(&f->id_table) == -1)
4974 goto out_free_name_table;
4975
4976 fuse_mutex_init(&f->lock);
4977
4978 root = alloc_node(f);
4979 if (root == NULL) {
4980 fuse_log(FUSE_LOG_ERR, "fuse: memory allocation failed\n");
4981 goto out_free_id_table;
4982 }
4983 if (lru_enabled(f)) {
4984 struct node_lru *lnode = node_lru(root);
4985 init_list_head(&lnode->lru);
4986 }
4987
4988 strcpy(root->inline_name, "/");
4989 root->name = root->inline_name;
4990
4991 if (f->conf.intr &&
4992 fuse_init_intr_signal(f->conf.intr_signal,
4993 &f->intr_installed) == -1)
4994 goto out_free_root;
4995
4996 root->parent = NULL;
4997 root->nodeid = FUSE_ROOT_ID;
4998 inc_nlookup(root);
4999 hash_id(f, root);
5000
5001 return f;
5002
5003 out_free_root:
5004 free(root);
5005 out_free_id_table:
5006 free(f->id_table.array);
5007 out_free_name_table:
5008 free(f->name_table.array);
5009 out_free_session:
5010 fuse_session_destroy(f->se);
5011 out_free_fs:
5012 if (f->fs->m)
5013 fuse_put_module(f->fs->m);
5014 free(f->fs);
5015 free(f->conf.modules);
5016 out_delete_context_key:
5017 fuse_delete_context_key();
5018 out_free:
5019 free(f);
5020 out:
5021 return NULL;
5022 }
5023
5024 /* Emulates 3.0-style fuse_new(), which processes --help */
5025 struct fuse *fuse_new_30(struct fuse_args *args, const struct fuse_operations *op,
5026 size_t op_size, void *private_data);
5027 FUSE_SYMVER(".symver fuse_new_30,fuse_new@FUSE_3.0");
fuse_new_30(struct fuse_args * args,const struct fuse_operations * op,size_t op_size,void * user_data)5028 struct fuse *fuse_new_30(struct fuse_args *args,
5029 const struct fuse_operations *op,
5030 size_t op_size, void *user_data)
5031 {
5032 struct fuse_config conf;
5033
5034 memset(&conf, 0, sizeof(conf));
5035
5036 const struct fuse_opt opts[] = {
5037 FUSE_LIB_OPT("-h", show_help, 1),
5038 FUSE_LIB_OPT("--help", show_help, 1),
5039 FUSE_OPT_END
5040 };
5041
5042 if (fuse_opt_parse(args, &conf, opts,
5043 fuse_lib_opt_proc) == -1)
5044 return NULL;
5045
5046 if (conf.show_help) {
5047 fuse_lib_help(args);
5048 return NULL;
5049 } else
5050 return fuse_new_31(args, op, op_size, user_data);
5051 }
5052
fuse_destroy(struct fuse * f)5053 void fuse_destroy(struct fuse *f)
5054 {
5055 size_t i;
5056
5057 if (f->conf.intr && f->intr_installed)
5058 fuse_restore_intr_signal(f->conf.intr_signal);
5059
5060 if (f->fs) {
5061 fuse_create_context(f);
5062
5063 for (i = 0; i < f->id_table.size; i++) {
5064 struct node *node;
5065
5066 for (node = f->id_table.array[i]; node != NULL;
5067 node = node->id_next) {
5068 if (node->is_hidden) {
5069 char *path;
5070 if (try_get_path(f, node->nodeid, NULL, &path, NULL, false) == 0) {
5071 fuse_fs_unlink(f->fs, path);
5072 free(path);
5073 }
5074 }
5075 }
5076 }
5077 }
5078 for (i = 0; i < f->id_table.size; i++) {
5079 struct node *node;
5080 struct node *next;
5081
5082 for (node = f->id_table.array[i]; node != NULL; node = next) {
5083 next = node->id_next;
5084 free_node(f, node);
5085 f->id_table.use--;
5086 }
5087 }
5088 assert(list_empty(&f->partial_slabs));
5089 assert(list_empty(&f->full_slabs));
5090
5091 while (fuse_modules) {
5092 fuse_put_module(fuse_modules);
5093 }
5094 free(f->id_table.array);
5095 free(f->name_table.array);
5096 pthread_mutex_destroy(&f->lock);
5097 fuse_session_destroy(f->se);
5098 free(f->conf.modules);
5099 free(f);
5100 fuse_delete_context_key();
5101 }
5102
fuse_mount(struct fuse * f,const char * mountpoint)5103 int fuse_mount(struct fuse *f, const char *mountpoint) {
5104 return fuse_session_mount(fuse_get_session(f), mountpoint);
5105 }
5106
5107
fuse_unmount(struct fuse * f)5108 void fuse_unmount(struct fuse *f) {
5109 fuse_session_unmount(fuse_get_session(f));
5110 }
5111
fuse_version(void)5112 int fuse_version(void)
5113 {
5114 return FUSE_VERSION;
5115 }
5116
fuse_pkgversion(void)5117 const char *fuse_pkgversion(void)
5118 {
5119 return PACKAGE_VERSION;
5120 }
5121