fuse整理

2024-02-23 14:18
文章标签 整理 fuse

本文主要是介绍fuse整理,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

FUSE 的工作原理如图所示。假设基于 FUSE 的用户态文件系统 hello 挂载在 /tmp/fuse 目录下。当应用层程序要访问 /tmp/fuse 下的文件时,通过 glibc 中的函数进行系统调用,处理这些系统调用的 VFS 中的函数会调用 FUSE 在内核中的文件系统;内核中的 FUSE 文件系统将用户的请求,发送给用户态文件系统 hello ;用户态文件系统收到请求后,进行处理,将结果返回给内核中的 FUSE 文件系统;最后,内核中的 FUSE 文件系统将数据返回给用户态程序。

FUSE整理 - bpingchang - bpingchang的博客

其中,用户态文件系统通过fuse提供的函数库libfuse,向内核中的fuse文件系统注册自己的文件处理函数。

我们以一个简单的基于FUSE的用户态文件系统hello为例,来分析FUSE的内部实现。 用 户态文件系统hello的源码hello.c的部分如下所示:

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_operationslibfuse提供给用户层文件系统的机 构,供用户定义自己的文件操作函数;fuse_main()libfuse提供给用户文件系统最重要的接口,通过这个函数,用户层文件系统可以将自己定义的fuse_operation注册为文件系 统的处理函数,并挂载该文件系统。

1、内核FUSE文件系统和用户态文件系统的通信

在加载fuse模块的过程中,要在内核中注册fuse文件系统,并生成fuse设备/dev/fuse

/dev/fuse是内核里的fuse文件系统和用户态文件系统的通信媒 介。用户态文件系统通过读取/dev/fuse的内容,获取内核中fuse文件系统发来的请求;而内核中的fuse文件系统,则把请求写入/dev/fuse,等待用户态文件系统处理。

在用户态文件系统和内核中的fuse模块通过/dev/fuse进行通信时,不可避免的会有内存拷贝。fuse中没有使用传统的copy_from_user/copy_to_user,而是用get_user_pagesmemcpy来代替。

2、FUSE用户态文件系统的挂载过程

挂载用户态文件系统,只需运行其生成的程序。在hello中,运行编译hello.c生成的hello即可。fuse_main将完成一切的注册和挂载过程,其中主要分为两个步骤:

2.1          在内核中挂载新的用户态文件系统

1.         打开设备文件"/dev/fuse" 获得文件描述符fd

2.         挂载FUSE文件系统(内核里的 fuse文件系统),将fd作为参数传给内核里的挂载函数

3.         内核中的fuse文件系统在初始化 super_block的过程中,将创建一个新的fuse_conn,它就是内核fuse和用户态文件系统通信的工具

4.         将该fuse_conn设置为在用户 态打开的/dev/fuse的私有数据(file->private);同时,该文件系统的super_blocks_fs_info也指向该 fuse_conn

2.2 注册用户态文件系统的处理函数,并创建进程处理该文件系统的请求

1.         创建一个新的fuse_chan,用 来与内核中的fuse通信。它的处理函数是 fuse_chan_ops

 

.receive = fuse_kern_chan_receive, 等待并从4.2.1-1步骤中打开的文件中读取请求

.send = fuse_kern_chan_send,            将用户态文件系统处理的结果发送给内 fuse

.destroy = fuse_kern_chan_destroy, 释放fuse_chan

 

2.         创建一个fuse_session,并 注册用户态文件系统的处理函数,用来等待并处理该文件系统的请求

3.         等待用户态文件系统的请求,并处理它 

 

3、多个基于FUSE的用户态文件系统的共存

在挂载的过程中,每个用户态文件系统 都会创建一个自己的fuse_conn,并将该fuse_conn设为自己打开的/dev/fuse的私有数据(file_private);同时,该文 件系统在内核里的super_block,也将该fuse_conn设置为s_fs_info。在以后的过程中,用户态的文件系统处理线程都是从自己的 fuse_conn上读取请求;内核中的fuse文件系统,也只把请求发给s_fs_info指向的fuse_conn。通过这种机制,FUSE就能支持 多个用户态文件系统同时运行,且互不干扰。

FUSE是一个用户空间中的文件系统框架。它包含2个部分,FUSE内核模块和用户空间接口模块。在用户空间的执行文件系统能够大幅提高生长率,简化了为操作系统提供新的文件系统的工作量,适合于各种虚拟文件系 统和网络文件系统。

FUSE 作为用户态文件系统与内核交互的桥梁,拥有着2个模块。其一是与用户态文件系统直接交互模块,如上图中的libfuse接口。其二是内核中挂载到VFS上的FUSE内核模块。

FUSE信息处理流程:

当用户发起一个请求ls –l /tmp/fuse 时,操作系统调用内核函数接口glibc将该请求发送到内核态VFSVFS根据请求判断出需要调用的文件系统(FUSE已经挂到VFS的文件系统列表上)并将此请求发送到FUSE内核模块;由内核模块再将此请求发 送到特殊文件/dev/fuse的请求队列中;而用户态的libfuse则不停的循环请求/dev/fuse文件中的请求队列,如获得请求则解析该请求,并发送给挂在其上的用户态文件系统,由它来执行请求。

在用户态文件系统执行完该请求后,按照逆方向返回执行结果给用户。

4用户态文件系统的挂载过程

1)用户态文件系统启动的第一部就是将自身注册到VFS上去。首先调用./lib/helper.c文件中函数fuse_mount(const char *mountpoint, struct fuse_args *args), 参数包括挂载点mountpoint等,在挂载过程中只需要输入挂载点路径即可。

2)通过调用./lib/helper.c文件中函数fuse_mount_compat25(const char *mountpoint, struct fuse_args *args) 再调用./lib/mount.c文件中的fuse_kern_mount(const char *mountpoint, struct fuse_args *args),接着调用static int fuse_mount_sys(const char *mnt, struct mount_opts *mo, const char *mnt_opts)函数,其中通过语句 fd = open(devname, O_RDWR);来检测文件/dev/fuse是否能够正常打开。如果正常,则直接调用系统函数res = mount(source, mnt, type, mo->flags, mo->kernel_opts);将其挂载到VFS上。

    3)创建并初始化一个跟/dev/fuse交互的通道结构体

struct fuse_chan {

struct fuse_chan_ops op; //其中操作包括receive, send, distroy三中方法。

struct fuse_session *se;  //通道的会话

int fd;      //文件/dev/fuse fd

size_t bufsize;     //pagesize() + 0x1000

void *data;

int compat;

};

//通道操作的动作绑定结构体

struct fuse_chan_ops op = {

           .receive = fuse_kern_chan_receive,

           .send = fuse_kern_chan_send,

           .destroy = fuse_kern_chan_destroy,

    };

最后将建立的通道fuse_chan返回给用户文件系统的main()函数。

5 FUSE 用户态模块的业务逻辑结构

1)        在用户态文件系统的main()函数中调用文件fuse_loop.c中的函数void fuse_session_add_chan(struct fuse_session *se, struct fuse_chan *ch),将通道地址赋值给该会话结构体struct fuse_session se, 然后通过调用同一文件中的函数int fuse_session_loop(struct fuse_session *se)来做信息的接收与发送工作。

struct fuse_session {

       struct fuse_session_ops op;  //会话的操作

       void *data;

       volatile int exited;

       struct fuse_chan *ch;

};

2)  在函数int fuse_chan_recv(struct fuse_chan **chp, char *buf, size_t size)中调用结构体struct fuse_chan 中绑定的操作函数static int fuse_kern_chan_receive(struct fuse_chan **chp, char *buf, size_t size)直接调用read()系统函数res = read(fuse_chan_fd(ch), buf, size); 来从/dev/fuse中读取消息。

//通道操作 的动作绑定结构体

struct fuse_chan_ops op = {

              .receive = fuse_kern_chan_receive,

              .send = fuse_kern_chan_send,

              .destroy = fuse_kern_chan_destroy,

};

3)   在函数void fuse_session_process(struct fuse_session *se, const char *buf, size_t len, struct fuse_chan *ch)中调用已绑定的se->op.process()函 数转向文件./lib/fuse_lowlevel.c中被绑定函数static void fuse_ll_process(void *data, const char *buf, size_t len, struct fuse_chan *ch)

struct fuse_session_ops sop = {

              .process = fuse_ll_process,

              .destroy = fuse_ll_destroy,

};

在这个函数处理过程中,首先要申明一个请求结构体指针struct fuse_req *req, 然 后通过在read(fuse_chan_fd(ch), buf, size); 函数执行中读取的数据buf,且将其强制转化为fuse_in_header类型(struct fuse_in_header *in = (struct fuse_in_header *) buf;),并初始化请求结构体req\

struct fuse_req {

}

通过解析出in->opcode的数据和用户态文 件系统定义的已绑定的接口函数相比较,得到相应的处理函数,最后执行这个函数完成请求。示例如下:如in->opcode FUSE_MKDIR 则通过结构体

6 FUSE 内核模块的业务逻辑结构

FUSE 作为用户态文件系统与内核交互的桥梁,拥有着2个模块。其一是与用户态文件系统直接交互模块,如上图中的libfuse接口。其二是内核中挂载到VFS上的FUSE内核模块。

FUSE信息处理流程:

当用户发起一个请求ls –l /tmp/fuse 时,操作系统调用内核函数接口glibc将该请求发送到内核态VFSVFS根据请求判断出需要调用的文件系统(FUSE已经挂到VFS的文件系统列表上)并将此请求发送到FUSE内核模块;由内核模块再将此请求发 送到特殊文件/dev/fuse的请求队列中;而用户态的libfuse则不停的循环请求/dev/fuse文件中的请求队列,如获得请求则解析该请求,并发送给挂在其上的用户态文件系统,由它来执行请求。

在用户态文件系统执行完该请求后,按照逆方向返回执行结果给用户。

6.1        用户态文件系统的挂载过程

1)        用户态文件系统启动的第一部就是将自身注册 到VFS上去。首 先调用./lib/helper.c文件中函数fuse_mount(const char *mountpoint, struct fuse_args *args), 参数包括挂载点mountpoint等,在挂载过程中只需 要输入挂载点路径即可。

2)        通过调用./lib/helper.c文件中函数fuse_mount_compat25(const char *mountpoint, struct fuse_args *args)再调用./lib/mount.c文件中的fuse_kern_mount(const char *mountpoint, struct fuse_args *args),接着调用static int fuse_mount_sys(const char *mnt, struct mount_opts *mo, const char *mnt_opts)函数,其中通过语句 fd = open(devname, O_RDWR);来检测文件/dev/fuse是否能够正常打开。如果正常,则直接调用系统函数res = mount(source, mnt, type, mo->flags, mo->kernel_opts);将其挂载到VFS上。

3)        创建并初始化一个跟/dev/fuse交互的通道结构体

struct fuse_chan {

struct fuse_chan_ops op; //其中操作包括receive, send, distroy三中方法。

struct fuse_session *se;  //通道的会话

int fd;                //文件/dev/fuse fd

size_t bufsize;         //pagesize() + 0x1000

void *data;

int compat;

};

//通道操作的动作绑定结构体

struct fuse_chan_ops op = {

           .receive = fuse_kern_chan_receive,

           .send = fuse_kern_chan_send,

           .destroy = fuse_kern_chan_destroy,

    };

最后将建立的通道fuse_chan返回给用户文件系统的main()函数。

6.2        FUSE用户态模块的业务逻辑结构

1)        在用户态文件系统的main()函数中调用文件fuse_loop.c中的函数void fuse_session_add_chan(struct fuse_session *se, struct fuse_chan *ch),将通道地址赋值给该会话结构体struct fuse_session se, 然后通过调用同一文件中的函数int fuse_session_loop(struct fuse_session *se)来做信息的接收与发送工作。

struct fuse_session {

       struct fuse_session_ops op;  //会话的操作

       void *data;

       volatile int exited;

       struct fuse_chan *ch;

};

接下来在while()循环中函数int fuse_chan_recv(struct fuse_chan **chp, char *buf, size_t size)用于消息的接收函数,如果接收到消息,则调用fuse_session.c文件中void fuse_session_process(struct fuse_session *se, const char *buf, size_t len,处理函数进行处理,其中调用会话控制结构体中的process函数指针所指向的函数来做实 际的处理(这个函数是由用户态文件系统提供的文件操作函数接口)。

struct fuse_session_ops {

void (*process) (void *data, const char *buf, size_t len,  struct fuse_chan *ch);

void (*exit) (void *data, int val);

int (*exited) (void *data);

void (*destroy) (void *data);

};

1)        在函数int fuse_chan_recv(struct fuse_chan **chp, char *buf, size_t size)中调用结构体struct fuse_chan中绑定的 操作函数static int fuse_kern_chan_receive(struct fuse_chan **chp, char *buf, size_t size)直接调用read()系统函数res = read(fuse_chan_fd(ch), buf, size); 来从/dev/fuse中读取消息。

//通道操作的动作绑定结构体

struct fuse_chan_ops op = {

              .receive = fuse_kern_chan_receive,

              .send = fuse_kern_chan_send,

              .destroy = fuse_kern_chan_destroy,

};

1)        在函数void fuse_session_process(struct fuse_session *se, const char *buf, size_t len, struct fuse_chan *ch)中 调用已绑定的se->op.process()函数转向文件./lib/fuse_lowlevel.c中被绑定函数static void fuse_ll_process(void *data, const char *buf, size_t len, struct fuse_chan *ch)

struct fuse_session_ops sop = {

              .process = fuse_ll_process,

              .destroy = fuse_ll_destroy,

};

在这个函数处理过程中,首先要申明一个请求结构体指针struct fuse_req *req, 然后通过在read(fuse_chan_fd(ch), buf, size); 函数执行中读取的数据buf,且将其强制转化为fuse_in_header类型(struct fuse_in_header *in = (struct fuse_in_header *) buf;),并初 始化请求结构体req\

struct fuse_req {}

通过解析出in->opcode的数据和用户态文件系统定义的已绑定的接口函数相比较,得到相应的处理函数,最后执行这个函数完成 请求。示例如下:如in->opcode  FUSE_MKDIR 则通过结构体

static struct {

       void (*func)(fuse_req_t, fuse_ino_t, const void *);

       const char *name;

}中已经初始化绑定的[FUSE_MKDIR] = { do_mkdir, "MKDIR" },一行数据找到函数static void do_mkdir(fuse_req_t req, fuse_ino_t nodeid, const void *inarg)最后来实现请求。

1)        当需要返回数据时,需要调用文件./lib/fuse_lowlevel.c中 的函数static int send_reply(fuse_req_t req, int error, const void *arg, size_t argsize),之后经过一系列的解析和变换,最后调用 文件./fuse_kern_chan.c中的static int fuse_kern_chan_send(struct fuse_chan *ch, const struct iovec iov[], size_t count)函数,它再调用系 统函数writev()ssize_t res = writev(fuse_chan_fd(ch), iov, count);)来将返回数据写入到/dev/fuse中去。

 

7FUSE内核模块的业务逻辑结构

1)        说明:(VFSFUSE内核模块的交互部分是由Linux内核编译生成的,暂时无法打印调试,因此对这部分代码的分析还不够确切,在接下来的工作中将会编译FUSE内核部分代码,并再次详细分析其运行过程)。

2)        当用户发出请求,若请求为:”rm /mnt/fuse/file” 时, 操作系统直接调用VFS函数sys_unlink()

1)        通过VFS的转向,调用已注册的“FUSE文件系统”(FUSE内核模块)的函数fuse_unlink()

2)        FUSE内核模块将接收到的请求保存到结构体struct fuse_conn fcfc->unused_list中,通 过函数request_send()函数将其写入fc->pending,同时使用函数fuse_dev_writev()将此请求写到特殊文件/dev/fuse中。

8、内核中/dev/fuse读写详细流程

1)        前提介绍

VFS要下发命令到FUSE或者从FUSE获取恢复时,首先找到与FUSE绑定的接口函数结构体(在dev.c中)。

struct file_operations fuse_dev_operations = {}

1)        读取/dev/fuse中的请求

在内核中要读取/dev/fuse中的请求数据时,最终要调用文件dev.c中的函数static ssize_t fuse_dev_readv(struct file *file, const struct iovec *iov, unsigned long nr_segs, loff_t *off) 此函数首先通过传入的参数file来初始化一个struct fuse_conn *fc结构体。然后调用请求等待函数static void request_wait(struct fuse_conn *fc), 先申明并初始化一个进程等待列表DECLARE_WAITQUEUE(wait, current); 

wait加入到fc->waitq等待队列中,当有请求发到fuse文件系统时(通过request_send),这个等待队列上的进程会被唤醒,某一个进程会被赋予CPU使用权:add_wait_queue_exclusive(&fc->waitq, &wait)接着不断的检查fcpending队列及interrupts队列,看是否有请求,没有请求会一直while循环。

while (fc->connected && !request_pending(fc)) {}

如果有请求,则break出来将当前进程状态设置为TASK_RUNNING状态,set_current_state(TASK_RUNNING);并将其从等待队列中移出,remove_wait_queue(&fc->waitq, &wait)  

fc->pending列表为非空时,证明已经有数据到达,那么作一些相应的状态修改,初始化等工作后,判断这个请求是否 合法,如果是合法请求,这从fc->pending列表中移除并拷贝其请求数据到用户空间的缓冲区。如果该请求是不需要回复的、意外终止、出现错误或者 正常结束的情况下调用函数static void request_end(struct fuse_conn *fc, struct fuse_req *req)来作一些处理后结束本次请求读取过程。

1)        将请求发送到/dev/fuse

首先申明一个结构体fc struct fuse_conn *fc = fuse_get_conn(file); 然后检查当前的设备状态,如 果没问题则初始化一个拷贝状态。

static void fuse_copy_init(struct fuse_copy_state *cs, struct fuse_conn *fc,

然后在当前的处理列表中查找写缓冲区的请求。如果找到,然后从列表中删除它并复制缓冲区的其余部 分。完成该请求通过调用函数static void request_end(struct fuse_conn *fc, struct fuse_req *req)

9、总结

FUSE内核模块主要工作就是进行队列的管 理,对fuse设 备的读(写)其实就是从相应的队列移除(添加)请求(或响应),request_send将请求加入pending队列,唤醒fuse守护程序,并在reqwaitq上等待请求结果,守护程序通过fuse_dev_readvpending队列中移除请求并处理,处理完成后,守护程序唤醒reqwaitq上的进程,该进程读取结果,并返回 给用户。

总的来说,一个请求从发起到完成会经过4步:

1)        fuse守护程序在fcwaitq上等待请求;

2)        用户的请求唤醒fcwaitq,从该waitq上移除一个请求进行处理,并在reqwaitq上等待请求结果;

3)        fuse守护程序被唤醒,读取请求,处理请求,返回结果,唤醒对应req上的waitq队列。

4)        请求被唤醒,读取fuse守护程序返回的结果,返回给用户。

这篇关于fuse整理的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/738913

相关文章

数论入门整理(updating)

一、gcd lcm 基础中的基础,一般用来处理计算第一步什么的,分数化简之类。 LL gcd(LL a, LL b) { return b ? gcd(b, a % b) : a; } <pre name="code" class="cpp">LL lcm(LL a, LL b){LL c = gcd(a, b);return a / c * b;} 例题:

rtmp流媒体编程相关整理2013(crtmpserver,rtmpdump,x264,faac)

转自:http://blog.163.com/zhujiatc@126/blog/static/1834638201392335213119/ 相关资料在线版(不定时更新,其实也不会很多,也许一两个月也不会改) http://www.zhujiatc.esy.es/crtmpserver/index.htm 去年在这进行rtmp相关整理,其实内容早有了,只是整理一下看着方

笔记整理—内核!启动!—kernel部分(2)从汇编阶段到start_kernel

kernel起始与ENTRY(stext),和uboot一样,都是从汇编阶段开始的,因为对于kernel而言,还没进行栈的维护,所以无法使用c语言。_HEAD定义了后面代码属于段名为.head .text的段。         内核起始部分代码被解压代码调用,前面关于uboot的文章中有提到过(eg:zImage)。uboot启动是无条件的,只要代码的位置对,上电就工作,kern

JavaScript整理笔记

JavaScript笔记 JavaScriptJavaScript简介快速入门JavaScript用法基础语法注释关键字显示数据输出innerHTML innerText属性返回值的区别调试 数据类型和变量数据类型数字(Number)字符串(String)布尔值(Boolean)null(空值)和undefined(未定义)数组(Array)对象(Object)函数(Function) 变量

关于回调函数和钩子函数基础知识的整理

回调函数:Callback Function 什么是回调函数? 首先做一个形象的比喻:   你有一个任务,但是有一部分你不会做,或者说不愿做,所以我来帮你做这部分,你做你其它的任务工作或者等着我的消息,但是当我完成的时候我要通知你我做好了,你可以用了,我怎么通知你呢?你给我一部手机,让我做完后给你打电话,我就打给你了,你拿到我的成果加到你的工作中,继续完成其它的工作.这就叫回叫,手机

站长常用Shell脚本整理分享(全)

站长常用Shell脚本整理分享 站长常用Shell脚本整理分享1-10 站长常用Shell脚本整理分享11-20 站长常用Shell脚本整理分享21-30 站长常用Shell脚本整理分享31-40 站长常用Shell脚本整理分享41-50 站长常用Shell脚本整理分享51-59 长期更新

我自己常用的eclipse 快捷键整理

---------------- 我自己改的快捷键: 复制当前行单下一行  ctrl alt n   --------------------- 自带快捷键: 快速定位到一行  CTRL+L 向上(下)移动选中的行:ALT+UP/DOWN ARROW 删除行(Delete Line):CTRL+D CTRL + 1也很有用     ----------

C/C++ 网络聊天室在线聊天系统(整理重传)

知识点: TCP网络通信 服务端的流程: 1.创建socket套接字 2.给这个socket绑定一个端口号 3.给这个socket开启监听属性 4.等待客户端连接 5.开始通讯 6.关闭连接 解释: socket:类似于接口的东西,只有通过这个才能跟对应的电脑通信。 每一台电脑都有一个IP地址,一台电脑上有多个应用,每个应用都会有一个端口号。 socket一般分为两种类型,一种是通讯,一种是监听

20190315 把整理和培养自己当作一生的事业,而不是局限在找工作拿offer。

把整理和培养自己当作一生的事业,而不是局限在找工作拿offer,做有本事的人。 来东南读研半年了,明显感觉自己掌握的不过是书本知识级别的中上水平,垃圾收集器这些的只知道背面经,靠脑子硬记,缺乏整理和系统,一头浆糊。 现在一边做实训这个烂项目,一边刷面经,一边刷剑指offer,想投些大公司的实习,又觉得还没准备好,看着各 种面经,都能说个大概,但明显感觉到自己知识的不体系和不深入,**做的项目

数据库系统原理概念整理(备考)

基本概念 数据模型 描述数据的概念和工具 关系数据模型 用关系描述数据 数据模型 包含三个方面 结构 操作 约束 对应于 关系数据模型 关系(表) 关系代数 主外键约束,断言 逻辑数据模型:详尽的描述数据,不关心具体的物理层实现,如关系数据模型中,设计实体及实体间的关系,属性,约束等等。业务逻辑的体现。 逻辑模型 --------查询处理----------物理模型 逻辑方面:SQL结构化查询