1

いくつかの問題があり、これの解決策が見つかりません:/ ヒューズ ファイルシステム内のテキスト ファイルを開く必要があります。デバッグではすべて正常に動作しますが、リリースではシステムがクラッシュしています。これで簡単な例を作りました。このコードの何が問題なのか、誰か教えていただけますか?

/*
  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);
}
4

2 に答える 2

0

わかりました、解決策を見つけました。パスは絶対ファイル パスである必要があります (これが適切な文かどうかはわかりません)。ただし、リリースとデバッグで動作するサンプル コードを次に示します。

/*
  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);
}
于 2011-04-01T12:15:39.307 に答える
0

readdir の実装には、少なくとも 2 つの問題があります。

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

于 2011-04-01T11:47:51.630 に答える