为什么FUSE似乎锁定了所有线程?

时间:2015-05-21 17:44:42

标签: c++ c fuse

我拿了保险丝hello.c并修改了底部以显示我在说什么。在我的应用程序中,我需要在保险丝FS可用后做一些事情。我还需要IPC的另一个线程,并保持某些事情是最新的。因为fuse_main似乎没有返回,所以我把它扔进了自己的主题。

当我评论fuse_main时,控制台显示A和B已打印。但是,如果我没有注释fuse_main(在不同的主题中),则仅打印A.如何保险丝停止我的主线程以及如何在FUSE执行此操作后运行代码?

#define FUSE_USE_VERSION 26

#include <fuse.h>
#include <stdio.h>
#include <string.h>
#include <errno.h>
#include <fcntl.h>

static const char *hello_str = "Hello World!\n";
static const char *hello_path = "/hello";

static int hello_getattr(const char *path, struct stat *stbuf)
{
    int res = 0;

    memset(stbuf, 0, sizeof(struct stat));
    if (strcmp(path, "/") == 0) {
        stbuf->st_mode = S_IFDIR | 0755;
        stbuf->st_nlink = 2;
    } else if (strcmp(path, hello_path) == 0) {
        stbuf->st_mode = S_IFREG | 0444;
        stbuf->st_nlink = 1;
        stbuf->st_size = strlen(hello_str);
    } else
        res = -ENOENT;

    return res;
}

static int hello_readdir(const char *path, void *buf, fuse_fill_dir_t filler,
             off_t offset, struct fuse_file_info *fi)
{
    (void) offset;
    (void) fi;

    if (strcmp(path, "/") != 0)
        return -ENOENT;

    filler(buf, ".", NULL, 0);
    filler(buf, "..", NULL, 0);
    filler(buf, hello_path + 1, NULL, 0);

    return 0;
}

static int hello_open(const char *path, struct fuse_file_info *fi)
{
    if (strcmp(path, hello_path) != 0)
        return -ENOENT;

    if ((fi->flags & 3) != O_RDONLY)
        return -EACCES;

    return 0;
}

static int hello_read(const char *path, char *buf, size_t size, off_t offset,
              struct fuse_file_info *fi)
{
    size_t len;
    (void) fi;
    if(strcmp(path, hello_path) != 0)
        return -ENOENT;

    len = strlen(hello_str);
    if (offset < len) {
        if (offset + size > len)
            size = len - offset;
        memcpy(buf, hello_str + offset, size);
    } else
        size = 0;

    return size;
}

static struct fuse_operations hello_oper;
//modification starts below this line
#include<thread>
#include<unistd.h>
int main(int argc, char *argv[])
{
    std::thread t([&]{
        hello_oper.getattr  = hello_getattr;
        hello_oper.readdir  = hello_readdir;
        hello_oper.open     = hello_open;
        hello_oper.read     = hello_read;

        return fuse_main(argc, argv, &hello_oper, NULL);
    });
    printf("A\n");
    sleep(5);
    printf("B\n");
    t.join();
}

2 个答案:

答案 0 :(得分:8)

fuse_main守护进程,即调用fork()并在父进程中调用_exit(0),以便进程退出,因此您只能看到A打印输出。

如果您在-f中提供选项./hello -f /tmp/fuse,则fuse_main不会调用_exit,但会在前台保持活跃状态​​A和{{1}可以看到。

当你的程序要退出时,你肯定需要一种方法来优雅地结束B线程:

fuse_main

//modification starts below this line #include<thread> #include<unistd.h> #include <signal.h> #include <sys/syscall.h> int main(int argc, char *argv[]) { pid_t tid; std::thread t([&]{ hello_oper.getattr = hello_getattr; hello_oper.readdir = hello_readdir; hello_oper.open = hello_open; hello_oper.read = hello_read; tid = syscall(SYS_gettid); return fuse_main(argc, argv, &hello_oper, NULL); }); printf("A\n"); sleep(5); printf("B\n"); kill(tid, SIGTERM); t.join(); } 的选项:

hello

答案 1 :(得分:4)

根据我在http://fuse.sourceforge.net/doxygen/hello_8c.html程序的文档中所读到的关于hello.c程序用法的内容,它说该程序退出并消失在后台,即{{1}的性质API。从这段代码开始,为什么不试一试 http://fuse.sourceforge.net/doxygen/hello__ll_8c.html,根据他们的描述, fuse_main这种方式,

unlike hello.c this example will stay in the foreground. it also replaced the convenience function fuse_main(..) with a more low level approach.

在主要功能中,你可以控制添加和做你想做的其他事情。

还有一个针对FUSE的c ++实现,https://code.google.com/p/fusekit/

希望这会有所帮助。