libfuse: выход из fuse_session_loop

Контекст: Ubuntu 11.10 и libfuse 2.8.4-1.4ubuntu1 Linux 3.0.0-14-generic # 23-Ubuntu SMP Понедельник, 21 ноября 20:28:43 UTC 2011 x86_64 x86_64 x86_64 GNU / Linux

Я пытаюсь использовать libfuse. Я хочу вызвать выход fuse_session_loop (из обработчика сигнала или другого потока), но когда я вызываю fuse_session_exit, ничего не происходит, пока сеанс не получит новый запрос.

fuse_session_exit устанавливает флаг, который читается fuse_session_exited. Отладка в fuse_session_loop, похоже, блокируется на fuse_chan_recv, поэтому он не проверяет fuse_session_exited снова до завершения цикла ...

int fuse_session_loop(struct fuse_session *se)
{
    int res = 0;
    struct fuse_chan *ch = fuse_session_next_chan(se, NULL);
    size_t bufsize = fuse_chan_bufsize(ch);
    char *buf = (char *) malloc(bufsize);
    if (!buf) {
        fprintf(stderr, "fuse: failed to allocate read buffer\n");
        return -1;
    }

    while (!fuse_session_exited(se)) {
        struct fuse_chan *tmpch = ch;
        res = fuse_chan_recv(&tmpch, buf, bufsize); <--- BLOCKING
        if (res == -EINTR)
            continue;
        if (res <= 0)
            break;
        fuse_session_process(se, buf, res, tmpch);
    }

    free(buf);
    fuse_session_reset(se);
    return res < 0 ? -1 : 0;
}

fuse_chan_recv вызывает fuse_kern_chan_receive, который блокирует системный вызов "read" устройства "/ dev / fuse" , поэтому даже если установлен флаг fuse_session_exited, ничего не происходит.

static int fuse_kern_chan_receive(struct fuse_chan **chp, char *buf,
                  size_t size)
{
    struct fuse_chan *ch = *chp;
    int err;
    ssize_t res;
    struct fuse_session *se = fuse_chan_session(ch);
    assert(se != NULL);

restart:
    res = read(fuse_chan_fd(ch), buf, size); <--- BLOCKING
    err = errno;

    if (fuse_session_exited(se))
        return 0;
    if (res == -1) {
        /* ENOENT means the operation was interrupted, it's safe
           to restart */
        if (err == ENOENT)
            goto restart;

        if (err == ENODEV) {
            fuse_session_exit(se);
            return 0;
        }
        /* Errors occuring during normal operation: EINTR (read
           interrupted), EAGAIN (nonblocking I/O), ENODEV (filesystem
           umounted) */
        if (err != EINTR && err != EAGAIN)
            perror("fuse: reading device");
        return -err;
    }
    if ((size_t) res < sizeof(struct fuse_in_header)) {
        fprintf(stderr, "short read on fuse device\n");
        return -EIO;
    }
    return res;
}

Эта проблема, похоже, связана с примером hello_ll.c, поставляемым с libfuse, а также с моей программой. Это заставляет меня думать, что, возможно, есть какой-то механизм, который не работает, и должен. Возможно, fuse_session_exit должен также делать что-то, что прерывает вызов чтения, что по какой-то причине не работает в моей системе.

Есть идеи?

5
задан Andrew Tomazos 18 January 2012 в 00:06
поделиться

0 ответов