libfuse__源码走读--多线程运行时

本文对源码的分析基于libfuse 3.12.0

一、fuse的多线程运行时

   fuse在运行时主要依靠fuse daemon与内核模块进行基于fuse协议的交互,libfuse通常采用多线程对这些请求进行处理,其中多线程运行时的主要实现在fuse_loop_mt.c中。

1.1 相关数据结构

//工作线程控制结构
struct fuse_worker {
	//双向循环链表
	struct fuse_worker *prev;
	struct fuse_worker *next;
	pthread_t thread_id;

	// We need to include fuse_buf so that we can properly free
	// it when a thread is terminated by pthread_cancel().
	struct fuse_buf fbuf;
	struct fuse_chan *ch;//与fuse内核连接的pipe的结构体
	struct fuse_mt *mt;//多线程控制结构
};
//多线程控制结构
struct fuse_mt {
	pthread_mutex_t lock;
	int numworker;//总线程数
	int numavail;//可用线程数
	struct fuse_session *se;//fuse会话结构体
	struct fuse_worker main;//工作主线程
	sem_t finish;//信号量
	int exit;
	int error;
	int clone_fd;//每个线程是否使用独立的设备fd
	int max_idle;//最大空闲线程数
	int max_threads;//最大线程数
};
//
struct fuse_chan {
	pthread_mutex_t lock;
	int ctr;//channel引用计数器
	int fd;//设备fd
};

1.2 从session启动函数看起

   fuse_session_loop_mt_31函数是进入fuse循环的主函数,该函数在用户进程和fuse内核模块的连接被中断时,会返回0。

int fuse_session_loop_mt_312(struct fuse_session *se, struct fuse_loop_config *config)
{
	int err;
	struct fuse_mt mt;
	struct fuse_worker *w;
	int created_config = 0;
	//检查fuse的config是否存在
	if (config) {	
		//检查config是否有效
		err = fuse_loop_cfg_verify(config);
		if (err)
			return err;
	} else {
		/* The caller does not care about parameters - use the default */
		config = fuse_loop_cfg_create();
		created_config = 1;
	}


	memset(&mt, 0, sizeof(struct fuse_mt));
	mt.se = se;
	mt.clone_fd = config->clone_fd;
	mt.error = 0;
	mt.numworker = 0;
	mt.numavail = 0;
	mt.max_idle = config->max_idle_threads;
	mt.max_threads = config->max_threads;
	mt.main.thread_id = pthread_self();
	mt.main.prev = mt.main.next = &mt.main;
	sem_init(&mt.finish, 0, 0);
	pthread_mutex_init(&mt.lock, NULL);

	pthread_mutex_lock(&mt.lock);
	err = fuse_loop_start_thread(&mt);//启动loop thread
	pthread_mutex_unlock(&mt.lock);
	if (!err) {
		/* sem_wait() is interruptible */
		while (!fuse_session_exited(se))//如果fuse会话没有退出,则阻塞在信号量上
			sem_wait(&mt.finish);

		pthread_mutex_lock(&mt.lock);
		for (w = mt.main.next; w != &mt.main; w = w->next)//终止所有任务线程
			pthread_cancel(w->thread_id);
		mt.exit = 1;
		pthread_mutex_unlock(&mt.lock);

		while (mt.main.next != &mt.main)
			fuse_join_worker(&mt, mt.main.next);//回收任务线程

		err = mt.error;
	}

	pthread_mutex_destroy(&mt.lock);
	sem_destroy(&mt.finish);
	if(se->error != 0)
		err = se->error;
	fuse_session_reset(se);

	if (created_config) {
		fuse_loop_cfg_destroy(config);
		config = NULL;
	}

	return err;
}

1.3 实际创建线程并执行event loop

static int fuse_loop_start_thread(struct fuse_mt *mt)
{
	int res;

	struct fuse_worker *w = malloc(sizeof(struct fuse_worker));
	if (!w) {
		fuse_log(FUSE_LOG_ERR, "fuse: failed to allocate worker structure\n");
		return -1;
	}
	memset(w, 0, sizeof(struct fuse_worker));
	w->fbuf.mem = NULL;
	w->mt = mt;

	w->ch = NULL;
	//检查是否需要为每个线程单独分配一个/dev/fuse的fd
	if (mt->clone_fd) {
		w->ch = fuse_clone_chan(mt);
		if(!w->ch) {
			/* Don't attempt this again */
			fuse_log(FUSE_LOG_ERR, "fuse: trying to continue "
				"without -o clone_fd.\n");
			mt->clone_fd = 0;
		}
	}

	res = fuse_start_thread(&w->thread_id, fuse_do_work, w);//调用该函数执行实际线程创建
	if (res == -1) {
		fuse_chan_put(w->ch);
		free(w);
		return -1;
	}
	list_add_worker(w, &mt->main);//添加到工作线程链表中
	mt->numavail ++;
	mt->numworker ++;

	return 0;
}

   在fuse_start_thread函数中会创建新线程,新线程执行fuse_do_work函数,这个函数才是实际执行event loop的函数。

static void *fuse_do_work(void *data)
{
	struct fuse_worker *w = (struct fuse_worker *) data;
	struct fuse_mt *mt = w->mt;

	while (!fuse_session_exited(mt->se)) {
		int isforget = 0;
		int res;

		pthread_setcancelstate(PTHREAD_CANCEL_ENABLE, NULL);
		res = fuse_session_receive_buf_int(mt->se, &w->fbuf, w->ch);//接收fuse内核传递的数据包,无数据读取时会阻塞住
		pthread_setcancelstate(PTHREAD_CANCEL_DISABLE, NULL);
		if (res == -EINTR)
			continue;
		if (res <= 0) {
			if (res < 0) {
				fuse_session_exit(mt->se);
				mt->error = res;
			}
			break;
		}

		pthread_mutex_lock(&mt->lock);
		if (mt->exit) {
			pthread_mutex_unlock(&mt->lock);
			return NULL;
		}

		/*
		 * This disgusting hack is needed so that zillions of threads
		 * are not created on a burst of FORGET messages
		 */
		if (!(w->fbuf.flags & FUSE_BUF_IS_FD)) {
			struct fuse_in_header *in = w->fbuf.mem;

			if (in->opcode == FUSE_FORGET ||
			    in->opcode == FUSE_BATCH_FORGET)
				isforget = 1;
		}

		if (!isforget)
			mt->numavail--;
		if (mt->numavail == 0 && mt->numworker < mt->max_threads)
			fuse_loop_start_thread(mt);//可用线程数小于最大线程数时创建新线程
		pthread_mutex_unlock(&mt->lock);
		//处理收到的数据包
		fuse_session_process_buf_int(mt->se, &w->fbuf, w->ch);

		pthread_mutex_lock(&mt->lock);
		if (!isforget)//处理完了,可用线程数+1
			mt->numavail++;

		/* creating and destroying threads is rather expensive - and there is
		 * not much gain from destroying existing threads. It is therefore
		 * discouraged to set max_idle to anything else than -1. If there
		 * is indeed a good reason to destruct threads it should be done
		 * delayed, a moving average might be useful for that.
		 */
		if (mt->max_idle != -1 && mt->numavail > mt->max_idle) {
			if (mt->exit) {
				pthread_mutex_unlock(&mt->lock);
				return NULL;
			}
			list_del_worker(w);
			mt->numavail--;
			mt->numworker--;
			pthread_mutex_unlock(&mt->lock);

			pthread_detach(w->thread_id);
			free(w->fbuf.mem);
			fuse_chan_put(w->ch);
			free(w);
			return NULL;
		}
		pthread_mutex_unlock(&mt->lock);
	}

	sem_post(&mt->finish);

	return NULL;
}

   其中接收buf主要通过res = read(ch ? ch->fd : se->fd, buf->mem, se->bufsize);,处理buf通过fuse_session_process_buf_int函数。处理buf的过程主要就是fuse数据协议的解析过程。该过程将会在后文讲解。

### 回答1: libfuse 是一个用户空间的文件系统开发库,它允许开发人员在不需要了解内核编程的情况下创建自己的文件系统。下面是一个简单的 libfuse 教程: 1. 安装 libfuse 首先,你需要安装 libfuse。如果你使用的是 Debian 或 Ubuntu 等基于 Debian 的发行版,可以使用以下命令进行安装: ``` sudo apt-get install libfuse-dev ``` 如果你使用的是 Fedora 或 Red Hat 等基于 Red Hat 的发行版,可以使用以下命令进行安装: ``` sudo dnf install fuse-devel ``` 2. 创建文件系统 接下来,你需要创建一个文件系统。在本例中,我们将创建一个名为 "hellofs" 的文件系统,该文件系统将在挂载点 /mnt/hello 中可用。在你的项目目录中创建一个名为 "hellofs.c" 的文件,并将以下代码复制到文件中: ```c #define FUSE_USE_VERSION 26 #include <fuse.h> #include <stdio.h> #include <string.h> #include <errno.h> #include <fcntl.h> 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") == 0) { stbuf->st_mode = S_IFREG | 0444; stbuf->st_nlink = 1; stbuf->st_size = strlen("Hello, World!\n"); } 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", NULL, 0); return 0; } static int hello_open(const char *path, struct fuse_file_info *fi) { if (strcmp(path, "/hello") != 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") != 0) return -ENOENT; len = strlen("Hello, World!\n"); if (offset < len) { if (offset + size > len) size = len - offset; memcpy(buf, "Hello, World!\n" + 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); } ``` 该代码实现了一个简单的文件系统,它包含一个名为 "hello" 的文件,该文件的内容为 "Hello, World!"。该文件系统的根目录包含一个名为 "hello" 的文件和 "." 和 ".." 目录。 3. 编译并运行文件系统 使用以下命令编译文件系统: ``` gcc -Wall hellofs.c `pkg-config fuse --cflags --libs` -o hellofs ``` 运行文件系统: ``` mkdir /mnt/hello ./hellofs /mnt/hello ``` 现在,你应该可以在 /mnt/hello 目录中看到 "hello" 文件。你可以尝试读取该文件,应该会输出 "Hello, World!"。 4. 卸载文件系统 要卸载文件系统,请使用以下命令: ``` fusermount -u /mnt/hello ``` 这就是一个简单的 libfuse 教程。你可以在此基础上进行扩展,创建你自己的文件系统。 ### 回答2: libfuse(Filesystem in Userspace)是一个开源的用户空间文件系统框架。它允许开发者在用户空间实现自己的文件系统,而不需要修改内核。下面是一个简要的libfuse教程: 1. 安装libfuse库:首先需要在机器上安装libfuse库。可以通过在终端中运行适当的安装命令来完成。例如,在Ubuntu上,可以运行`sudo apt-get install libfuse-dev`进行安装。 2. 编写文件系统代码:之后,需要编写自己的文件系统代码。这些代码定义了文件系统的各种操作,如读取、写入、创建、删除文件等。libfuse提供了一组API(应用程序编程接口),用于处理这些操作。 3. 编译和运行:编写完文件系统代码后,可以将其编译成可执行文件。在编译时,需要链接libfuse库。编译完成后,可以运行可执行文件,启动自己的文件系统。 4. 挂载文件系统:在运行文件系统之前,需要将其挂载到实际目录上,使其可以被访问和操作。可以使用`fusermount`命令来实现挂载。例如,可以运行`fusermount -u /mnt/myfilesystem`来卸载文件系统。 5. 测试和调试:一旦文件系统成功挂载,就可以测试和调试它了。可以尝试在挂载目录中创建、读取、写入文件,以及执行其他相关操作。同时,可以使用调试工具和日志来跟踪和解决可能的问题。 需要注意的是,libfuse不仅支持Linux系统,还可以在其他类Unix系统上使用。此外,libfuse还提供了一些选项,允许开发者自定义文件系统的行为和特性。 总结起来,libfuse提供了一个方便的方式来开发用户空间文件系统。通过按照上述步骤,开发者可以快速上手,并创建自己的文件系统。对于那些想要实现自定义文件系统的开发者来说,libfuse是一个强大且易于使用的工具。 ### 回答3: libfuse 是一个用于开发文件系统的库,它可以使开发者能够创建自定义的文件系统,而无需在内核中进行修改或编译操作。使用 libfuse,我们可以在用户空间中实现文件系统操作,这样就可以更加灵活地定制文件系统的功能和行为。 libfuse 提供了一系列的 API,可以用来注册文件系统的回调函数,并进行文件的读写、创建和删除等操作。开发者可以根据自己的需求,实现这些回调函数来处理文件系统的各种操作。 libfuse 教程主要介绍了如何使用 libfuse 来开发自定义的文件系统。教程首先介绍了安装 libfuse 和相关开发工具的步骤。然后详细讲解了如何编写一个最简单的文件系统,这个文件系统只有根目录,并能够列出目录中的文件和子目录。 接下来的章节逐步扩展了文件系统的功能,介绍了如何创建和删除文件、如何读写文件、如何处理文件权限等等。同时,教程中还提供了一些示例代码,可以帮助开发者更好地理解和使用 libfuselibfuse 教程的最后一部分还介绍了一些高级特性,如如何处理符号链接、如何实现权限控制和文件锁定等等。教程还介绍了 libfuse 的一些配置选项和常用命令行参数,帮助开发者更好地使用和管理文件系统。 总之,libfuse 教程为开发者提供了一个全面且易于理解的指南,帮助他们快速入门并掌握 libfuse 的使用技巧,从而能够更加灵活地开发自定义的文件系统。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值