#include <stddef.h>
#include <unistd.h>
#include <pthread.h>
+#include <stdbool.h>
+#include <stdatomic.h>
/* We can't actually tell the kernel that there is no
timeout, so we just send a big value */
static char file_contents[MAX_STR_LEN];
static int lookup_cnt = 0;
static size_t file_size;
+static _Atomic bool is_stop = false;
/* Command line parsing */
struct options {
return 0;
}
+static void tfs_destroy(void *userarg)
+{
+ (void)userarg;
+
+ is_stop = true;
+}
+
static void tfs_lookup(fuse_req_t req, fuse_ino_t parent,
const char *name) {
struct fuse_entry_param e;
}
static const struct fuse_lowlevel_ops tfs_oper = {
+ .destroy = tfs_destroy,
.lookup = tfs_lookup,
.getattr = tfs_getattr,
.readdir = tfs_readdir,
static void* update_fs_loop(void *data) {
struct fuse_session *se = (struct fuse_session*) data;
- while(1) {
+ while(!is_stop) {
update_fs();
if (!options.no_notify && lookup_cnt) {
- /* Only send notification if the kernel
- is aware of the inode */
- assert(fuse_lowlevel_notify_inval_inode
- (se, FILE_INO, 0, 0) == 0);
+ /* Only send notification if the kernel is aware of the inode */
+
+ /* Some errors (ENOENT, EBADFD, ENODEV) have to be accepted as the
+ * might come up during umount, when kernel side already releases
+ * all inodes, but does not send FUSE_DESTROY yet.
+ */
+ int ret =
+ fuse_lowlevel_notify_inval_inode(se, FILE_INO, 0, 0);
+ if ((ret != 0 && !is_stop) &&
+ ret != -ENOENT && ret != -EBADFD && ret != -ENODEV) {
+ fprintf(stderr,
+ "ERROR: fuse_lowlevel_notify_store() failed with %s (%d)\n",
+ strerror(-ret), -ret);
+ abort();
+ }
}
sleep(options.update_interval);
}