Сбой файловой системы FUSE при открытии файла (только в режиме выпуска) - PullRequest
1 голос
/ 01 апреля 2011

У меня есть проблема, и я не могу найти решение этой проблемы: / Мне нужно открыть текстовый файл внутри моей файловой системы fuse.В отладке все работает нормально, но в релизной системе происходит сбой.Я сделал простой пример с этим.Кто-нибудь может сказать, что не так в этом коде?

/*
  FUSE: Filesystem in Userspace
  Copyright (C) 2001-2007  Miklos Szeredi <miklos@szeredi.hu>

  This program can be distributed under the terms of the GNU GPL.
  See the file COPYING.

  gcc -Wall `pkg-config fuse --cflags --libs` hello.c -o hello
*/
#    define FUSE_USE_VERSION 26

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

void stripnl(char *str) {
  while(strlen(str) && ( (str[strlen(str) - 1] == 13) || 
       ( str[strlen(str) - 1] == 10 ))) {
    str[strlen(str) - 1] = 0;
  }
}

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);

  FILE *infile;
  char fname[40];
  char line[100];
  int lcount;

      /* We need to get rid of the newline char. */
  stripnl(fname);

  /* Open the file.  If NULL is returned there was an error */
  if((infile = fopen("ex.txt", "r")) == NULL) {
    printf("Error Opening File.\n");
  }

  while( fgets(line, sizeof(line), infile) != NULL ) {
    /* Get each line from the infile */
    lcount++;
    /* print the line number and data */
    printf("Line %d: %s", lcount, line);  
    filler(buf, line, NULL, 0);
  }

  fclose(infile);  /* Close the file */

    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 = {
    .getattr    = hello_getattr,
    .readdir    = hello_readdir,
    .open       = hello_open,
    .read       = hello_read,
};

int main(int argc, char *argv[])
{
    return fuse_main(argc, argv, &hello_oper, NULL);
}

Обновление

Хорошо, я нашел решение, путь должен быть абсолютным путем файла (не уверен, что это правильное предложение), но вот примеркод, который работает как в выпуске, так и в отладке:

/*
  FUSE: Filesystem in Userspace
  Copyright (C) 2001-2007  Miklos Szeredi <miklos@szeredi.hu>

  This program can be distributed under the terms of the GNU GPL.
  See the file COPYING.

  gcc -Wall `pkg-config fuse --cflags --libs` hello.c -o hello
*/
#    define FUSE_USE_VERSION 26

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

void stripnl(char *str) {
  while(strlen(str) && ( (str[strlen(str) - 1] == 13) || 
       ( str[strlen(str) - 1] == 10 ))) {
    str[strlen(str) - 1] = 0;
  }
}

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);

    FILE *infile;
    char line[100];

    if((infile = fopen("/root/Desktop/fexamples/ex.txt", "rb")) == NULL) 
    {
        return -1;
    }

    while( fgets(line, sizeof(line), infile) != NULL ) 
    {
            filler(buf, line, NULL, 0);
    }

    fclose(infile);  /* Close the file */

    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 = {
    .getattr    = hello_getattr,
    .readdir    = hello_readdir,
    .open       = hello_open,
    .read       = hello_read,
};

int main(int argc, char *argv[])
{
    return fuse_main(argc, argv, &hello_oper, NULL);
}

Ответы [ 2 ]

0 голосов
/ 01 апреля 2011

Хорошо, я нашел решение, путь должен быть абсолютным путем к файлу (не уверен, что это правильное предложение), но вот пример кода, который работает как в релизе, так и в отладке:

/*
  FUSE: Filesystem in Userspace
  Copyright (C) 2001-2007  Miklos Szeredi <miklos@szeredi.hu>

  This program can be distributed under the terms of the GNU GPL.
  See the file COPYING.

  gcc -Wall `pkg-config fuse --cflags --libs` hello.c -o hello
*/
#    define FUSE_USE_VERSION 26

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

void stripnl(char *str) {
  while(strlen(str) && ( (str[strlen(str) - 1] == 13) || 
       ( str[strlen(str) - 1] == 10 ))) {
    str[strlen(str) - 1] = 0;
  }
}

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);

    FILE *infile;
    char line[100];

    if((infile = fopen("/root/Desktop/fexamples/ex.txt", "rb")) == NULL) 
    {
        return -1;
    }

    while( fgets(line, sizeof(line), infile) != NULL ) 
    {
            filler(buf, line, NULL, 0);
    }

    fclose(infile);  /* Close the file */

    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 = {
    .getattr    = hello_getattr,
    .readdir    = hello_readdir,
    .open       = hello_open,
    .read       = hello_read,
};

int main(int argc, char *argv[])
{
    return fuse_main(argc, argv, &hello_oper, NULL);
}
0 голосов
/ 01 апреля 2011

В вашей реализации readdir есть как минимум две проблемы.

char fname[40];
char line[100];
int lcount;

/* We need to get rid of the newline char. */
stripnl(fname);

Вы действуете с содержимым fname, которое вы нигде не инициализируете.Это может сделать что угодно.Тот факт, что вы не используете fname после этого, не имеет значения, вы могли бы изменять совершенно случайные данные с помощью этого вызова stripnl.

/* Open the file.  If NULL is returned there was an error */
if((infile = fopen("ex.txt", "r")) == NULL) {
    printf("Error Opening File.\n");
}

Если fopen завершится неудачно, вы должны выполнить ошибкуили просто уйти) эта функция.В противном случае следующая строка будет вызывать fgets с NULL в качестве аргумента файла.

...