examples/notify_store_retrieve: Add a clean shutdown
authorBernd Schubert <bschubert@ddn.com>
Tue, 26 Dec 2023 10:53:55 +0000 (11:53 +0100)
committerNikolaus Rath <Nikolaus@rath.org>
Thu, 28 Dec 2023 09:18:12 +0000 (09:18 +0000)
On shutdown update_fs_loop() was failing frequently in github
test with ENOENT, but only ENODEV was tested for and only for the.
With clean shutdown there is no need to test for such errors at all.

example/notify_store_retrieve.c

index 0a165c59305e4796d28fb104763b1abde4765bfa..c4d00876ffc2b388a793db29965ccf95421d76b8 100644 (file)
@@ -70,6 +70,7 @@
 #include <stddef.h>
 #include <unistd.h>
 #include <pthread.h>
+#include <stdbool.h>
 
 /* We can't actually tell the kernel that there is no
    timeout, so we just send a big value */
@@ -86,6 +87,12 @@ static size_t file_size;
    received it back correctly (==2) */
 static int retrieve_status = 0;
 
+static bool is_umount = false;
+
+/* updater thread tid */
+static pthread_t updater;
+
+
 /* Command line parsing */
 struct options {
     int no_notify;
@@ -268,6 +275,15 @@ static void tfs_retrieve_reply(fuse_req_t req, void *cookie, fuse_ino_t ino,
     fuse_reply_none(req);
 }
 
+static void tfs_destroy(void *userdata)
+{
+       (void)userdata;
+
+       is_umount = true;
+
+       pthread_join(updater, NULL);
+}
+
 
 static const struct fuse_lowlevel_ops tfs_oper = {
     .lookup    = tfs_lookup,
@@ -277,6 +293,7 @@ static const struct fuse_lowlevel_ops tfs_oper = {
     .read      = tfs_read,
     .forget     = tfs_forget,
     .retrieve_reply = tfs_retrieve_reply,
+    .destroy    = tfs_destroy,
 };
 
 static void update_fs(void) {
@@ -296,7 +313,7 @@ static void* update_fs_loop(void *data) {
     struct fuse_bufvec bufv;
     int ret;
 
-    while(1) {
+    while(!is_umount) {
         update_fs();
         if (!options.no_notify && lookup_cnt) {
             /* Only send notification if the kernel
@@ -311,11 +328,7 @@ static void* update_fs_loop(void *data) {
             /* This shouldn't fail, but apparently it sometimes
                does - see https://github.com/libfuse/libfuse/issues/105 */
             ret = fuse_lowlevel_notify_store(se, FILE_INO, 0, &bufv, 0);
-            if (-ret == ENODEV) {
-                // File system was unmounted
-                break;
-            }
-            else if (ret != 0) {
+            if (ret != 0 && !is_umount) {
                 fprintf(stderr, "ERROR: fuse_lowlevel_notify_store() failed with %s (%d)\n",
                         strerror(-ret), -ret);
                 abort();
@@ -325,10 +338,7 @@ static void* update_fs_loop(void *data) {
                kernel to send us back the stored data */
             ret = fuse_lowlevel_notify_retrieve(se, FILE_INO, MAX_STR_LEN,
                                                 0, (void*) strdup(file_contents));
-            if (-ret == ENODEV) { // File system was unmounted
-                break;
-            }
-            assert(ret == 0);
+            assert(ret == 0 || is_umount);
             if(retrieve_status == 0)
                 retrieve_status = 1;
         }
@@ -351,7 +361,6 @@ int main(int argc, char *argv[]) {
     struct fuse_session *se;
     struct fuse_cmdline_opts opts;
     struct fuse_loop_config config;
-    pthread_t updater;
     int ret = -1;
 
     if (fuse_opt_parse(&args, &options, option_spec, NULL) == -1)