Disable old symbol versions if __UCLIBC__ is defined fuse_2_7_2_before_indent
authorMiklos Szeredi <miklos@szeredi.hu>
Wed, 12 Dec 2007 11:53:38 +0000 (11:53 +0000)
committerMiklos Szeredi <miklos@szeredi.hu>
Wed, 12 Dec 2007 11:53:38 +0000 (11:53 +0000)
17 files changed:
ChangeLog
configure.in
kernel/configure.ac
lib/Makefile.am
lib/fuse.c
lib/fuse_lowlevel.c
lib/fuse_misc.h
lib/fuse_mt.c
lib/fuse_opt.c
lib/fuse_session.c
lib/fuse_versionscript
lib/helper.c
lib/mount.c
lib/mount_bsd.c
test/.cvsignore
test/stracedecode.c [new file with mode: 0644]
test/test.c

index 5cdbc9ad9eab1af25aaa84506d253d5245d99ce6..d292db77eb616c7f885b15ccfb8981f0aa097d67 100644 (file)
--- a/ChangeLog
+++ b/ChangeLog
@@ -1,3 +1,16 @@
+2007-12-12  Miklos Szeredi <miklos@szeredi.hu>
+
+       * Disable old symbol versions if __UCLIBC__ is defined.  If a
+       symbol in a library has multiple versions, the runtime linker in
+       uClibc seems to randomly choose between them.
+
+       * Remove erroneous 'fuse_opt_insert_arg@FUSE_2_5' from
+       fuse_version_script.  fuse_opt_free_args() was added in fuse-2.6.
+
+       * Close fuse device file descriptor before calling umount(),
+       preventing a deadlock when umount is synchronous.  Reported by
+       Szabolcs Szakacsits
+
 2007-11-12  Miklos Szeredi <miklos@szeredi.hu>
 
        * 'fusermount -u' did not umount the filesystem if /etc/mtab was a
index f81dfea9f5846978acfe1cc17e94f7b5fd18970c..d1f0fcf221fea4d95a58ef26c313d6a09757a83a 100644 (file)
@@ -1,4 +1,4 @@
-AC_INIT(fuse, 2.7.1)
+AC_INIT(fuse, 2.7.2)
 AC_CANONICAL_TARGET
 AM_INIT_AUTOMAKE
 AM_CONFIG_HEADER(include/config.h)
index 489a1e1418030d23d3d1b085caca8f0af884fc37..ab7b09cd83ba870732667923c99ca2bb5fa6312d 100644 (file)
@@ -1,4 +1,4 @@
-AC_INIT(fuse-kernel, 2.7.1)
+AC_INIT(fuse-kernel, 2.7.2)
 AC_CONFIG_HEADERS([config.h])
 
 AC_PROG_INSTALL
index cf848471607e42c8fae89b1ba2555f2fd8169519..4d14ef47aa1984032bd2761b52958ba511472a5d 100644 (file)
@@ -34,7 +34,7 @@ libfuse_la_SOURCES =          \
        $(iconv_source)         \
        $(mount_source)
 
-libfuse_la_LDFLAGS = @libfuse_libs@ -version-number 2:7:1 \
+libfuse_la_LDFLAGS = @libfuse_libs@ -version-number 2:7:2 \
        -Wl,--version-script,$(srcdir)/fuse_versionscript
 
 libulockmgr_la_SOURCES = ulockmgr.c
index 80bfe60360bbc517c13ed78d6a1947d9640e6b2c..d22e200e3f18f2de82eb96f55e758e4de46ec510 100644 (file)
@@ -3290,12 +3290,12 @@ struct fuse *fuse_new_compat1(int fd, int flags,
                                   sizeof(struct fuse_operations_compat1), 11);
 }
 
-__asm__(".symver fuse_exited,__fuse_exited@");
-__asm__(".symver fuse_process_cmd,__fuse_process_cmd@");
-__asm__(".symver fuse_read_cmd,__fuse_read_cmd@");
-__asm__(".symver fuse_set_getcontext_func,__fuse_set_getcontext_func@");
-__asm__(".symver fuse_new_compat2,fuse_new@");
-__asm__(".symver fuse_new_compat22,fuse_new@FUSE_2.2");
+FUSE_SYMVER(".symver fuse_exited,__fuse_exited@");
+FUSE_SYMVER(".symver fuse_process_cmd,__fuse_process_cmd@");
+FUSE_SYMVER(".symver fuse_read_cmd,__fuse_read_cmd@");
+FUSE_SYMVER(".symver fuse_set_getcontext_func,__fuse_set_getcontext_func@");
+FUSE_SYMVER(".symver fuse_new_compat2,fuse_new@");
+FUSE_SYMVER(".symver fuse_new_compat22,fuse_new@FUSE_2.2");
 
 #endif /* __FreeBSD__ */
 
@@ -3307,4 +3307,4 @@ struct fuse *fuse_new_compat25(int fd, struct fuse_args *args,
                                     op_size, 25);
 }
 
-__asm__(".symver fuse_new_compat25,fuse_new@FUSE_2.5");
+FUSE_SYMVER(".symver fuse_new_compat25,fuse_new@FUSE_2.5");
index 64e2aba0989f2305ae38056dca661c0bc1e9e0e5..d39a4a1c172e88f3ef0322ce88c5ebb6d5731e56 100644 (file)
@@ -1409,9 +1409,9 @@ int fuse_sync_compat_args(struct fuse_args *args)
     return 0;
 }
 
-__asm__(".symver fuse_reply_statfs_compat,fuse_reply_statfs@FUSE_2.4");
-__asm__(".symver fuse_reply_open_compat,fuse_reply_open@FUSE_2.4");
-__asm__(".symver fuse_lowlevel_new_compat,fuse_lowlevel_new@FUSE_2.4");
+FUSE_SYMVER(".symver fuse_reply_statfs_compat,fuse_reply_statfs@FUSE_2.4");
+FUSE_SYMVER(".symver fuse_reply_open_compat,fuse_reply_open@FUSE_2.4");
+FUSE_SYMVER(".symver fuse_lowlevel_new_compat,fuse_lowlevel_new@FUSE_2.4");
 
 #else /* __FreeBSD__ */
 
@@ -1435,4 +1435,4 @@ struct fuse_session *fuse_lowlevel_new_compat25(struct fuse_args *args,
                                     op_size, userdata);
 }
 
-__asm__(".symver fuse_lowlevel_new_compat25,fuse_lowlevel_new@FUSE_2.5");
+FUSE_SYMVER(".symver fuse_lowlevel_new_compat25,fuse_lowlevel_new@FUSE_2.5");
index 99e5117ae601d28d14b37f0d0bdc30e8d0d273c3..4f5236a4d18d3fbbfcdcc68435ed14c39255ca7c 100644 (file)
@@ -9,9 +9,17 @@
 #include "config.h"
 #include <pthread.h>
 
+/* Versioned symbols confuse the dynamic linker in uClibc */
+#ifndef __UCLIBC__
+#define FUSE_SYMVER(x) __asm__(x)
+#else
+#define FUSE_SYMVER(x)
+#endif
+
 #ifndef USE_UCLIBC
 #define fuse_mutex_init(mut) pthread_mutex_init(mut, NULL)
 #else
+/* Is this hack still needed? */
 static inline void fuse_mutex_init(pthread_mutex_t *mut)
 {
     pthread_mutexattr_t attr;
index 2c1ff2cfee260abb584a8a57c92a5141b8eb2509..cbdc1a316c77e3085ff5e52171af4f73954ab934 100644 (file)
@@ -7,6 +7,7 @@
 */
 
 #include "fuse_i.h"
+#include "fuse_misc.h"
 #include "fuse_lowlevel.h"
 
 #include <stdio.h>
@@ -112,4 +113,4 @@ int fuse_loop_mt(struct fuse *f)
     return fuse_session_loop_mt(fuse_get_session(f));
 }
 
-__asm__(".symver fuse_loop_mt_proc,__fuse_loop_mt@");
+FUSE_SYMVER(".symver fuse_loop_mt_proc,__fuse_loop_mt@");
index 1c6a3157b745f744a18b77bfddc40d2e1a3c2f78..d10e62468f9c7ed2e141e0a6fd9c7ce0f4feb4eb 100644 (file)
@@ -7,6 +7,7 @@
 */
 
 #include "fuse_opt.h"
+#include "fuse_misc.h"
 
 #include <stdio.h>
 #include <stdlib.h>
@@ -65,7 +66,8 @@ int fuse_opt_add_arg(struct fuse_args *args, const char *arg)
     return 0;
 }
 
-int fuse_opt_insert_arg(struct fuse_args *args, int pos, const char *arg)
+static int fuse_opt_insert_arg_common(struct fuse_args *args, int pos,
+                                     const char *arg)
 {
     assert(pos <= args->argc);
     if (fuse_opt_add_arg(args, arg) == -1)
@@ -80,6 +82,18 @@ int fuse_opt_insert_arg(struct fuse_args *args, int pos, const char *arg)
     return 0;
 }
 
+int fuse_opt_insert_arg(struct fuse_args *args, int pos, const char *arg)
+{
+       return fuse_opt_insert_arg_common(args, pos, arg);
+}
+
+int fuse_opt_insert_arg_compat(struct fuse_args *args, int pos,
+                              const char *arg);
+int fuse_opt_insert_arg_compat(struct fuse_args *args, int pos, const char *arg)
+{
+       return fuse_opt_insert_arg_common(args, pos, arg);
+}
+
 static int next_arg(struct fuse_opt_context *ctx, const char *opt)
 {
     if (ctx->argctr + 1 >= ctx->argc) {
@@ -365,3 +379,6 @@ int fuse_opt_parse(struct fuse_args *args, void *data,
     fuse_opt_free_args(&ctx.outargs);
     return res;
 }
+
+/* This symbol version was mistakenly added to the version script */
+FUSE_SYMVER(".symver fuse_opt_insert_arg_compat,fuse_opt_insert_arg@FUSE_2.5");
index 33cc59e86bcd00af60ca0c45018ea7a13ff715ae..cf2e20b19a22ee569cd73c26ef0b5f3ce90f6a98 100644 (file)
@@ -7,6 +7,7 @@
 */
 
 #include "fuse_lowlevel.h"
+#include "fuse_misc.h"
 #include "fuse_common_compat.h"
 #include "fuse_lowlevel_compat.h"
 
@@ -204,5 +205,5 @@ void fuse_chan_destroy(struct fuse_chan *ch)
 }
 
 #ifndef __FreeBSD__
-__asm__(".symver fuse_chan_new_compat24,fuse_chan_new@FUSE_2.4");
+FUSE_SYMVER(".symver fuse_chan_new_compat24,fuse_chan_new@FUSE_2.4");
 #endif
index 9d6642a52ef340bd4e55a0021247494fd96a8b3a..3eedd0c6ddbcf911feef98bd4d34cc4e1398021a 100644 (file)
@@ -67,7 +67,6 @@ FUSE_2.5 {
                fuse_opt_add_opt;
                fuse_opt_add_arg;
                fuse_opt_free_args;
-               fuse_opt_insert_arg;
                fuse_opt_match;
                fuse_parse_cmdline;
                fuse_remove_signal_handlers;
index e79f4b017a9bca7534d0c5c2534530aeb7b5da15..fbbf4b26f0820289cc096c2d2d4fded94978eb9b 100644 (file)
@@ -8,6 +8,7 @@
 
 #include "config.h"
 #include "fuse_i.h"
+#include "fuse_misc.h"
 #include "fuse_opt.h"
 #include "fuse_lowlevel.h"
 #include "fuse_common_compat.h"
@@ -402,11 +403,11 @@ int fuse_mount_compat1(const char *mountpoint, const char *args[])
     return fuse_mount_compat22(mountpoint, NULL);
 }
 
-__asm__(".symver fuse_setup_compat2,__fuse_setup@");
-__asm__(".symver fuse_setup_compat22,fuse_setup@FUSE_2.2");
-__asm__(".symver fuse_teardown,__fuse_teardown@");
-__asm__(".symver fuse_main_compat2,fuse_main@");
-__asm__(".symver fuse_main_real_compat22,fuse_main_real@FUSE_2.2");
+FUSE_SYMVER(".symver fuse_setup_compat2,__fuse_setup@");
+FUSE_SYMVER(".symver fuse_setup_compat22,fuse_setup@FUSE_2.2");
+FUSE_SYMVER(".symver fuse_teardown,__fuse_teardown@");
+FUSE_SYMVER(".symver fuse_main_compat2,fuse_main@");
+FUSE_SYMVER(".symver fuse_main_real_compat22,fuse_main_real@FUSE_2.2");
 
 #endif /* __FreeBSD__ */
 
@@ -439,7 +440,7 @@ int fuse_mount_compat25(const char *mountpoint, struct fuse_args *args)
     return fuse_kern_mount(mountpoint, args);
 }
 
-__asm__(".symver fuse_setup_compat25,fuse_setup@FUSE_2.5");
-__asm__(".symver fuse_teardown_compat22,fuse_teardown@FUSE_2.2");
-__asm__(".symver fuse_main_real_compat25,fuse_main_real@FUSE_2.5");
-__asm__(".symver fuse_mount_compat25,fuse_mount@FUSE_2.5");
+FUSE_SYMVER(".symver fuse_setup_compat25,fuse_setup@FUSE_2.5");
+FUSE_SYMVER(".symver fuse_teardown_compat22,fuse_teardown@FUSE_2.2");
+FUSE_SYMVER(".symver fuse_main_real_compat25,fuse_main_real@FUSE_2.5");
+FUSE_SYMVER(".symver fuse_mount_compat25,fuse_mount@FUSE_2.5");
index e0fea56b234d44bf97d179ca8b7008120ca8da92..8a5c5abe8ac8402bec25545a29b32ab65c1e2815 100644 (file)
@@ -8,6 +8,7 @@
 
 #include "config.h"
 #include "fuse_i.h"
+#include "fuse_misc.h"
 #include "fuse_opt.h"
 #include "fuse_common_compat.h"
 #include "mount_util.h"
@@ -283,6 +284,10 @@ void fuse_kern_unmount(const char *mountpoint, int fd)
            then the filesystem is already unmounted */
         if (res == 1 && (pfd.revents & POLLERR))
             return;
+
+        /* Need to close file descriptor, otherwise synchronous umount
+           would recurse into filesystem, and deadlock */
+        close(fd);
     }
 
     if (geteuid() == 0) {
@@ -579,5 +584,5 @@ int fuse_kern_mount(const char *mountpoint, struct fuse_args *args)
     return res;
 }
 
-__asm__(".symver fuse_mount_compat22,fuse_mount@FUSE_2.2");
-__asm__(".symver fuse_unmount_compat22,fuse_unmount@FUSE_2.2");
+FUSE_SYMVER(".symver fuse_mount_compat22,fuse_mount@FUSE_2.2");
+FUSE_SYMVER(".symver fuse_unmount_compat22,fuse_unmount@FUSE_2.2");
index 1c4030d221c066f0a43b96cebca362b833ed7511..5197464d7666b38762055fdf3f5d6e0717280ea9 100644 (file)
@@ -7,6 +7,7 @@
 */
 
 #include "fuse_i.h"
+#include "fuse_misc.h"
 #include "fuse_opt.h"
 
 #include <sys/stat.h>
@@ -360,4 +361,4 @@ int fuse_kern_mount(const char *mountpoint, struct fuse_args *args)
     return res;
 }
 
-__asm__(".symver fuse_unmount_compat22,fuse_unmount@FUSE_2.2");
+FUSE_SYMVER(".symver fuse_unmount_compat22,fuse_unmount@FUSE_2.2");
index 9daeafb9864cf43055ae93beb0afd6c7d144bfa4..efb4fee1a16f02681087ec103eda1dedec290d2f 100644 (file)
@@ -1 +1,2 @@
 test
+stracedecode
diff --git a/test/stracedecode.c b/test/stracedecode.c
new file mode 100644 (file)
index 0000000..7908f43
--- /dev/null
@@ -0,0 +1,197 @@
+#include <stdio.h>
+#include <string.h>
+#include "fuse_kernel.h"
+
+static struct {
+    const char *name;
+} fuse_ll_ops[] = {
+    [FUSE_LOOKUP]      = { "LOOKUP"      },
+    [FUSE_FORGET]      = { "FORGET"      },
+    [FUSE_GETATTR]     = { "GETATTR"     },
+    [FUSE_SETATTR]     = { "SETATTR"     },
+    [FUSE_READLINK]    = { "READLINK"    },
+    [FUSE_SYMLINK]     = { "SYMLINK"     },
+    [FUSE_MKNOD]       = { "MKNOD"       },
+    [FUSE_MKDIR]       = { "MKDIR"       },
+    [FUSE_UNLINK]      = { "UNLINK"      },
+    [FUSE_RMDIR]       = { "RMDIR"       },
+    [FUSE_RENAME]      = { "RENAME"      },
+    [FUSE_LINK]        = { "LINK"        },
+    [FUSE_OPEN]        = { "OPEN"        },
+    [FUSE_READ]        = { "READ"        },
+    [FUSE_WRITE]       = { "WRITE"       },
+    [FUSE_STATFS]      = { "STATFS"      },
+    [FUSE_RELEASE]     = { "RELEASE"     },
+    [FUSE_FSYNC]       = { "FSYNC"       },
+    [FUSE_SETXATTR]    = { "SETXATTR"    },
+    [FUSE_GETXATTR]    = { "GETXATTR"    },
+    [FUSE_LISTXATTR]   = { "LISTXATTR"   },
+    [FUSE_REMOVEXATTR] = { "REMOVEXATTR" },
+    [FUSE_FLUSH]       = { "FLUSH"       },
+    [FUSE_INIT]        = { "INIT"        },
+    [FUSE_OPENDIR]     = { "OPENDIR"     },
+    [FUSE_READDIR]     = { "READDIR"     },
+    [FUSE_RELEASEDIR]  = { "RELEASEDIR"  },
+    [FUSE_FSYNCDIR]    = { "FSYNCDIR"    },
+    [FUSE_GETLK]       = { "GETLK"       },
+    [FUSE_SETLK]       = { "SETLK"       },
+    [FUSE_SETLKW]      = { "SETLKW"      },
+    [FUSE_ACCESS]      = { "ACCESS"      },
+    [FUSE_CREATE]      = { "CREATE"      },
+    [FUSE_INTERRUPT]   = { "INTERRUPT"   },
+    [FUSE_BMAP]        = { "BMAP"        },
+    [FUSE_DESTROY]     = { "DESTROY"     },
+};
+
+#define FUSE_MAXOP (sizeof(fuse_ll_ops) / sizeof(fuse_ll_ops[0]))
+
+static const char *opname(enum fuse_opcode opcode)
+{
+    if (opcode >= FUSE_MAXOP || !fuse_ll_ops[opcode].name)
+        return "???";
+    else
+        return fuse_ll_ops[opcode].name;
+}
+
+
+static void process_buf(int dir, char *buf, int len)
+{
+    static unsigned long long prevuniq = -1;
+    static int prevopcode;
+
+    if (!dir) {
+        struct fuse_in_header *in = (struct fuse_in_header *) buf;
+        buf += sizeof(struct fuse_in_header);
+
+        printf("unique: %llu, opcode: %s (%i), nodeid: %lu, len: %i, insize: %i\n",
+               (unsigned long long) in->unique,
+               opname((enum fuse_opcode) in->opcode), in->opcode,
+               (unsigned long) in->nodeid, in->len, len);
+
+        switch (in->opcode) {
+        case FUSE_READ: {
+            struct fuse_read_in *arg = (struct fuse_read_in *) buf;
+            printf("-READ fh:%llu off:%llu siz:%u rfl:%u own:%llu fl:%u\n",
+                   arg->fh, arg->offset, arg->size, arg->read_flags,
+                   arg->lock_owner, arg->flags);
+            break;
+        }
+        case FUSE_WRITE: {
+            struct fuse_write_in *arg = (struct fuse_write_in *) buf;
+            printf("-WRITE fh:%llu off:%llu siz:%u wfl:%u own:%llu fl:%u\n",
+                   arg->fh, arg->offset, arg->size, arg->write_flags,
+                   arg->lock_owner, arg->flags);
+            break;
+        }
+        }
+        prevuniq = in->unique;
+        prevopcode = in->opcode;
+    } else {
+        struct fuse_out_header *out = (struct fuse_out_header *) buf;
+        buf += sizeof(struct fuse_out_header);
+
+        printf("   unique: %llu, error: %i (%s), len: %i, outsize: %i\n",
+                (unsigned long long) out->unique, out->error,
+                strerror(-out->error), out->len, len);
+
+        if (out->unique == prevuniq) {
+            switch (prevopcode) {
+            case FUSE_GETATTR: {
+                struct fuse_attr_out *arg = (struct fuse_attr_out *) buf;
+                printf("+ATTR v:%llu.%09u i:%llu s:%llu b:%llu\n",
+                       arg->attr_valid, arg->attr_valid_nsec,
+                       arg->attr.ino, arg->attr.size, arg->attr.blocks);
+                break;
+            }
+            case FUSE_LOOKUP: {
+                struct fuse_entry_out *arg = (struct fuse_entry_out *) buf;
+                printf("+ENTRY nodeid:%llu v:%llu.%09u i:%llu s:%llu b:%llu\n",
+                       arg->nodeid, arg->attr_valid, arg->attr_valid_nsec,
+                       arg->attr.ino, arg->attr.size, arg->attr.blocks);
+                break;
+            }
+            }
+        }
+    }
+
+}
+
+int main(void)
+{
+    FILE *in = stdin;
+    while (1) {
+        int dir;
+        int res;
+        char buf[1048576];
+        unsigned len = 0;
+
+        memset(buf, 0, sizeof(buf));
+        while (1) {
+            char str[32];
+
+            res = fscanf(in, "%30s", str);
+            if (res != 1 && feof(in))
+                return 0;
+
+            if (res == 0)
+                continue;
+
+            if (strncmp(str, "read(", 5) == 0) {
+                dir = 0;
+                break;
+            } else if (strncmp(str, "writev(", 7) == 0) {
+                dir = 1;
+                break;
+            }
+        }
+
+        while (1) {
+            int c = getc(in);
+            if (c == '"') {
+                while (1) {
+                    int val;
+
+                    c = getc(in);
+                    if (c == EOF) {
+                        fprintf(stderr, "eof in string\n");
+                        break;
+                    }
+                    if (c == '\n') {
+                        fprintf(stderr, "eol in string\n");
+                        break;
+                    }
+                    if (c == '"')
+                        break;
+                    if (c != '\\') {
+                        val = c;
+                    } else {
+                        c = getc(in);
+                        switch (c) {
+                        case 'n': val = '\n'; break;
+                        case 'r': val = '\r'; break;
+                        case 't': val = '\t'; break;
+                        case '"': val = '"'; break;
+                        case '\\': val = '\\'; break;
+                        case 'x':
+                            res = scanf("%x", &val);
+                            if (res != 1) {
+                                fprintf(stderr, "parse error\n");
+                                continue;
+                            }
+                            break;
+                        default:
+                            fprintf(stderr, "unknown sequence: '\\%c'\n", c);
+                            continue;
+                        }
+                    }
+                    buf[len++] = val;
+                }
+            }
+            if (c == '\n')
+                break;
+        }
+        process_buf(dir, buf, len);
+        memset(buf, 0, len);
+        len = 0;
+    }
+}
index c1646d20497149cfebef00dbb7599a8f52bc5813..f9aa823158212ae7c57003191510753a738130e9 100644 (file)
@@ -1254,6 +1254,10 @@ int main(int argc, char *argv[])
     }
     basepath = argv[1];
     assert(strlen(basepath) < 512);
+    if (basepath[0] != '/') {
+        fprintf(stderr, "testdir must be an absolute path\n");
+        return 1;
+    }
 
     sprintf(testfile, "%s/testfile", basepath);
     sprintf(testfile2, "%s/testfile2", basepath);