receive和process的过程

(一)

receive最终在fuse_kern_chan.c中的fuse_kern_chan_receive函数实现,使用系统调用读取

        res = read(fuse_chan_fd(ch), buf, size);

其中fuse_chan_fd(ch)就是/dev/fuse, buf和size在fuse_loop_mt.c中fuse_do_work中定义。      

struct fuse_buf fbuf = {
  .mem = w->buf,
  .size = w->bufsize,
};

其中的fbuf->mem就是buf,fbuf->size就是size。

size的值为 135168

这些值在 fuse_loop_mt.c::fuse_loop_start_thread中被赋值。

w->bufsize = fuse_chan_bufsize(mt->prevch);

w->buf = malloc(w->bufsize); 

receive完成后,就是process。process函数最终是fuse_lowlevel.c中的fuse_ll_process_buf 。

 

-------process应该和receive类似,最终在fuse_kern_chan.c的fuse_kern_chan_send函数中调用 :

      ssize_t res = writev(fuse_chan_fd(ch), iov, count);

writev也是一个系统调用 http://man7.org/linux/man-pages/man2/readv.2.html

 

(二)

执行ls dir1命令,fuse_ll_process_buf 会收到这些操作码:

  3  FUSE_GETATTR

[getattr] Called
  Attributes of / requested

  27  FUSE_OPENDIR

  28  FUSE_READDIR

--> Getting The List of Files of /

  1  FUSE_LOOKUP

[getattr] Called
  Attributes of /file54 requested

  1

[getattr] Called
  Attributes of /file349 requested

  28  FUSE_READDIR

  29  FUSE_RELEASEDIR

  42  FUSE_BATCH_FORGET

每一个操作码都耗费fuse_do_work(fuse_lootp_mt.c)的一个循环。即一个receive,一个process。

最终按照操作码调用对应函数:   

    fuse_ll_ops[in->opcode].func(req, in->nodeid, inarg);

in根据fuse_ll_process_buf中的参数buf得来。

func函数也在fuse_lowlevel.c中,比如do_getattr函数,该函数调用:

     req->f->op.getattr(req, nodeid, fip);

去执行用户定义的函数。 这里的op类型是struct fuse_lowlevel_ops,定义在fuse_lowlevel.c。

最终应该会去往fuse.c中的fuse_fs_getattr函数,在里面调用fs->op.getattr(path, buf);这个getattr函数在用户程序ssfs.c中定义。

 

op是在用户程序ssfs.c中定义,由fuse_main函数传递而来。

 

(三)

由于用户程序ssfs.c中只定义了三种操作,所以只会接受到三种操作。没有定义的操作,在fuse_lowlevel.c中进行对应的判断如if (req->f->op.setxattr)不成立于是调用fuse_reply_err(req, ENOSYS);进行回应。

(四)接上面(二)的结尾

static struct fuse_lowlevel_ops fuse_path_ops 在fuse.c中定义,有一句 

  .getattr = fuse_lib_getattr,

全局搜索只有这一处比较像lowlevel的getattr函数指针。

在fuse.c中,有

  static void fuse_lib_getattr(fuse_req_t req, fuse_ino_t ino,struct fuse_file_info *fi)

这与fuse_lowlevel.c中的getattr函数里面的调用是一致的。

  fuse_lib_getattr函数调用fuse_fs_fgetattr(f->fs, path, &buf, fi);, 里面调用fs->op.getattr(path, buf);

        这里应该就是用户定义的getattr了。

这里fs->op正是struct fuse_operations类型的。这个变量由用户程序ssfs.c的fuse_main传入。

posted on 2017-04-26 22:18  longbigfish  阅读(350)  评论(0编辑  收藏  举报

导航