| 
									
										
										
										
											2022-06-03 19:15:11 +00:00
										 |  |  | /*   SPDX-License-Identifier: BSD-3-Clause
 | 
					
						
							| 
									
										
										
										
											2022-11-01 20:26:26 +00:00
										 |  |  |  *   Copyright (C) 2019 Intel Corporation. | 
					
						
							| 
									
										
										
										
											2019-09-25 08:47:02 +00:00
										 |  |  |  *   All rights reserved. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "spdk/stdinc.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "spdk/log.h"
 | 
					
						
							|  |  |  | #include "spdk/env.h"
 | 
					
						
							|  |  |  | #include "spdk/event.h"
 | 
					
						
							|  |  |  | #include "spdk/thread.h"
 | 
					
						
							|  |  |  | #include "spdk/string.h"
 | 
					
						
							|  |  |  | #include "spdk/blobfs.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "blobfs_fuse.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define FUSE_USE_VERSION 30
 | 
					
						
							|  |  |  | #include "fuse3/fuse.h"
 | 
					
						
							|  |  |  | #include "fuse3/fuse_lowlevel.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct spdk_blobfs_fuse { | 
					
						
							|  |  |  | 	char *bdev_name; | 
					
						
							|  |  |  | 	char *mountpoint; | 
					
						
							|  |  |  | 	struct spdk_fs_thread_ctx *channel; | 
					
						
							|  |  |  | 	struct spdk_filesystem *fs; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	struct fuse *fuse_handle; | 
					
						
							|  |  |  | 	pthread_t	fuse_tid; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	blobfs_fuse_unmount_cb cb_fn; | 
					
						
							|  |  |  | 	void *cb_arg; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Each thread serves one blobfs */ | 
					
						
							|  |  |  | static __thread struct spdk_blobfs_fuse *thd_bfuse; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | blobfs_fuse_free(struct spdk_blobfs_fuse *bfuse) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (bfuse == NULL) { | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	free(bfuse->bdev_name); | 
					
						
							|  |  |  | 	free(bfuse->mountpoint); | 
					
						
							|  |  |  | 	free(bfuse); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | __call_fn(void *arg1, void *arg2) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	fs_request_fn fn; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	fn = (fs_request_fn)arg1; | 
					
						
							|  |  |  | 	fn(arg2); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void | 
					
						
							| 
									
										
										
										
											2020-05-10 18:49:31 +00:00
										 |  |  | blobfs_fuse_send_request(fs_request_fn fn, void *arg) | 
					
						
							| 
									
										
										
										
											2019-09-25 08:47:02 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	struct spdk_event *event; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	event = spdk_event_allocate(0, __call_fn, (void *)fn, arg); | 
					
						
							|  |  |  | 	spdk_event_call(event); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							| 
									
										
										
										
											2020-05-10 18:49:31 +00:00
										 |  |  | fuse_getattr(const char *path, struct stat *stbuf, struct fuse_file_info *fi) | 
					
						
							| 
									
										
										
										
											2019-09-25 08:47:02 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	struct spdk_file_stat stat; | 
					
						
							|  |  |  | 	int rc; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!strcmp(path, "/")) { | 
					
						
							|  |  |  | 		stbuf->st_mode = S_IFDIR | 0755; | 
					
						
							|  |  |  | 		stbuf->st_nlink = 2; | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	rc = spdk_fs_file_stat(thd_bfuse->fs, thd_bfuse->channel, path, &stat); | 
					
						
							|  |  |  | 	if (rc == 0) { | 
					
						
							|  |  |  | 		stbuf->st_mode = S_IFREG | 0644; | 
					
						
							|  |  |  | 		stbuf->st_nlink = 1; | 
					
						
							|  |  |  | 		stbuf->st_size = stat.size; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return rc; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							| 
									
										
										
										
											2020-05-10 18:49:31 +00:00
										 |  |  | fuse_readdir(const char *path, void *buf, fuse_fill_dir_t filler, | 
					
						
							|  |  |  | 	     off_t offset, struct fuse_file_info *fi, | 
					
						
							|  |  |  | 	     enum fuse_readdir_flags flags) | 
					
						
							| 
									
										
										
										
											2019-09-25 08:47:02 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	struct spdk_file *file; | 
					
						
							|  |  |  | 	const char *filename; | 
					
						
							|  |  |  | 	spdk_fs_iter iter; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	filler(buf, ".", NULL, 0, 0); | 
					
						
							|  |  |  | 	filler(buf, "..", NULL, 0, 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	iter = spdk_fs_iter_first(thd_bfuse->fs); | 
					
						
							|  |  |  | 	while (iter != NULL) { | 
					
						
							|  |  |  | 		file = spdk_fs_iter_get_file(iter); | 
					
						
							|  |  |  | 		iter = spdk_fs_iter_next(iter); | 
					
						
							|  |  |  | 		filename = spdk_file_get_name(file); | 
					
						
							|  |  |  | 		filler(buf, &filename[1], NULL, 0, 0); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							| 
									
										
										
										
											2020-05-10 18:49:31 +00:00
										 |  |  | fuse_mknod(const char *path, mode_t mode, dev_t rdev) | 
					
						
							| 
									
										
										
										
											2019-09-25 08:47:02 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	return spdk_fs_create_file(thd_bfuse->fs, thd_bfuse->channel, path); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							| 
									
										
										
										
											2020-05-10 18:49:31 +00:00
										 |  |  | fuse_unlink(const char *path) | 
					
						
							| 
									
										
										
										
											2019-09-25 08:47:02 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	return spdk_fs_delete_file(thd_bfuse->fs, thd_bfuse->channel, path); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							| 
									
										
										
										
											2020-05-10 18:49:31 +00:00
										 |  |  | fuse_truncate(const char *path, off_t size, struct fuse_file_info *fi) | 
					
						
							| 
									
										
										
										
											2019-09-25 08:47:02 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	struct spdk_file *file; | 
					
						
							|  |  |  | 	int rc; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	rc = spdk_fs_open_file(thd_bfuse->fs, thd_bfuse->channel, path, 0, &file); | 
					
						
							|  |  |  | 	if (rc != 0) { | 
					
						
							|  |  |  | 		return -rc; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	rc = spdk_file_truncate(file, thd_bfuse->channel, size); | 
					
						
							|  |  |  | 	if (rc != 0) { | 
					
						
							|  |  |  | 		return -rc; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	spdk_file_close(file, thd_bfuse->channel); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							| 
									
										
										
										
											2020-05-10 18:49:31 +00:00
										 |  |  | fuse_utimens(const char *path, const struct timespec tv[2], struct fuse_file_info *fi) | 
					
						
							| 
									
										
										
										
											2019-09-25 08:47:02 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							| 
									
										
										
										
											2020-05-10 18:49:31 +00:00
										 |  |  | fuse_open(const char *path, struct fuse_file_info *info) | 
					
						
							| 
									
										
										
										
											2019-09-25 08:47:02 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	struct spdk_file *file; | 
					
						
							|  |  |  | 	int rc; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	rc = spdk_fs_open_file(thd_bfuse->fs, thd_bfuse->channel, path, 0, &file); | 
					
						
							|  |  |  | 	if (rc != 0) { | 
					
						
							|  |  |  | 		return -rc; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	info->fh = (uintptr_t)file; | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							| 
									
										
										
										
											2020-05-10 18:49:31 +00:00
										 |  |  | fuse_release(const char *path, struct fuse_file_info *info) | 
					
						
							| 
									
										
										
										
											2019-09-25 08:47:02 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	struct spdk_file *file = (struct spdk_file *)info->fh; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return spdk_file_close(file, thd_bfuse->channel); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							| 
									
										
										
										
											2020-05-10 18:49:31 +00:00
										 |  |  | fuse_read(const char *path, char *buf, size_t len, off_t offset, struct fuse_file_info *info) | 
					
						
							| 
									
										
										
										
											2019-09-25 08:47:02 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	struct spdk_file *file = (struct spdk_file *)info->fh; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return spdk_file_read(file, thd_bfuse->channel, buf, offset, len); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							| 
									
										
										
										
											2020-05-10 18:49:31 +00:00
										 |  |  | fuse_write(const char *path, const char *buf, size_t len, off_t offset, | 
					
						
							|  |  |  | 	   struct fuse_file_info *info) | 
					
						
							| 
									
										
										
										
											2019-09-25 08:47:02 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	struct spdk_file *file = (struct spdk_file *)info->fh; | 
					
						
							|  |  |  | 	int rc; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	rc = spdk_file_write(file, thd_bfuse->channel, (void *)buf, offset, len); | 
					
						
							|  |  |  | 	if (rc == 0) { | 
					
						
							|  |  |  | 		return len; | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		return rc; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							| 
									
										
										
										
											2020-05-10 18:49:31 +00:00
										 |  |  | fuse_flush(const char *path, struct fuse_file_info *info) | 
					
						
							| 
									
										
										
										
											2019-09-25 08:47:02 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							| 
									
										
										
										
											2020-05-10 18:49:31 +00:00
										 |  |  | fuse_fsync(const char *path, int datasync, struct fuse_file_info *info) | 
					
						
							| 
									
										
										
										
											2019-09-25 08:47:02 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							| 
									
										
										
										
											2020-05-10 18:49:31 +00:00
										 |  |  | fuse_rename(const char *old_path, const char *new_path, unsigned int flags) | 
					
						
							| 
									
										
										
										
											2019-09-25 08:47:02 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	return spdk_fs_rename_file(thd_bfuse->fs, thd_bfuse->channel, old_path, new_path); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct fuse_operations spdk_fuse_oper = { | 
					
						
							| 
									
										
										
										
											2020-05-10 18:49:31 +00:00
										 |  |  | 	.getattr	= fuse_getattr, | 
					
						
							|  |  |  | 	.readdir	= fuse_readdir, | 
					
						
							|  |  |  | 	.mknod		= fuse_mknod, | 
					
						
							|  |  |  | 	.unlink		= fuse_unlink, | 
					
						
							|  |  |  | 	.truncate	= fuse_truncate, | 
					
						
							|  |  |  | 	.utimens	= fuse_utimens, | 
					
						
							|  |  |  | 	.open		= fuse_open, | 
					
						
							|  |  |  | 	.release	= fuse_release, | 
					
						
							|  |  |  | 	.read		= fuse_read, | 
					
						
							|  |  |  | 	.write		= fuse_write, | 
					
						
							|  |  |  | 	.flush		= fuse_flush, | 
					
						
							|  |  |  | 	.fsync		= fuse_fsync, | 
					
						
							|  |  |  | 	.rename		= fuse_rename, | 
					
						
							| 
									
										
										
										
											2019-09-25 08:47:02 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void * | 
					
						
							|  |  |  | fuse_loop_new_thread(void *arg) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct spdk_blobfs_fuse *bfuse = arg; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	spdk_unaffinitize_thread(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	thd_bfuse = bfuse; | 
					
						
							|  |  |  | 	SPDK_NOTICELOG("Start to loop blobfs on bdev %s mounted at %s\n", bfuse->bdev_name, | 
					
						
							|  |  |  | 		       bfuse->mountpoint); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	bfuse->channel = spdk_fs_alloc_thread_ctx(bfuse->fs); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	fuse_loop(bfuse->fuse_handle); | 
					
						
							|  |  |  | 	fuse_unmount(bfuse->fuse_handle); | 
					
						
							|  |  |  | 	fuse_destroy(bfuse->fuse_handle); | 
					
						
							|  |  |  | 	SPDK_NOTICELOG("Blobfs on bdev %s unmounted from %s\n", bfuse->bdev_name, bfuse->mountpoint); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	spdk_fs_free_thread_ctx(bfuse->channel); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	bfuse->cb_fn(bfuse->cb_arg); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	blobfs_fuse_free(bfuse); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	pthread_exit(NULL); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int | 
					
						
							| 
									
										
										
										
											2020-05-10 18:49:31 +00:00
										 |  |  | blobfs_fuse_start(const char *bdev_name, const char *mountpoint, struct spdk_filesystem *fs, | 
					
						
							|  |  |  | 		  blobfs_fuse_unmount_cb cb_fn, void *cb_arg, struct spdk_blobfs_fuse **_bfuse) | 
					
						
							| 
									
										
										
										
											2019-09-25 08:47:02 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	/* Set argv[1] as bdev_name in order to show bdev_name as the mounting source */ | 
					
						
							|  |  |  | 	char *argv[1] = {(char *)bdev_name}; | 
					
						
							|  |  |  | 	struct fuse_args args = FUSE_ARGS_INIT(1, argv); | 
					
						
							|  |  |  | 	struct fuse_cmdline_opts opts = {}; | 
					
						
							|  |  |  | 	struct fuse *fuse_handle; | 
					
						
							|  |  |  | 	struct spdk_blobfs_fuse *bfuse; | 
					
						
							|  |  |  | 	pthread_t tid; | 
					
						
							|  |  |  | 	int rc; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	bfuse = (struct spdk_blobfs_fuse *)calloc(1, sizeof(*bfuse)); | 
					
						
							|  |  |  | 	if (bfuse == NULL) { | 
					
						
							|  |  |  | 		return -ENOMEM; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	bfuse->bdev_name = strdup(bdev_name); | 
					
						
							|  |  |  | 	bfuse->mountpoint = strdup(mountpoint); | 
					
						
							| 
									
										
										
										
											2021-06-13 10:25:14 +00:00
										 |  |  | 	if (!bfuse->bdev_name || !bfuse->mountpoint) { | 
					
						
							|  |  |  | 		rc = -ENOMEM; | 
					
						
							|  |  |  | 		goto err; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-09-25 08:47:02 +00:00
										 |  |  | 	bfuse->fs = fs; | 
					
						
							|  |  |  | 	bfuse->cb_fn = cb_fn; | 
					
						
							|  |  |  | 	bfuse->cb_arg = cb_arg; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-13 10:25:14 +00:00
										 |  |  | 	rc = fuse_parse_cmdline(&args, &opts); | 
					
						
							|  |  |  | 	assert(rc == 0); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-25 08:47:02 +00:00
										 |  |  | 	fuse_handle = fuse_new(&args, &spdk_fuse_oper, sizeof(spdk_fuse_oper), NULL); | 
					
						
							|  |  |  | 	fuse_opt_free_args(&args); | 
					
						
							|  |  |  | 	if (fuse_handle == NULL) { | 
					
						
							|  |  |  | 		SPDK_ERRLOG("could not create fuse handle!\n"); | 
					
						
							|  |  |  | 		rc = -1; | 
					
						
							|  |  |  | 		goto err; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	bfuse->fuse_handle = fuse_handle; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	rc = fuse_mount(bfuse->fuse_handle, bfuse->mountpoint); | 
					
						
							|  |  |  | 	if (rc != 0) { | 
					
						
							|  |  |  | 		SPDK_ERRLOG("could not mount fuse handle\n"); | 
					
						
							|  |  |  | 		rc = -1; | 
					
						
							|  |  |  | 		goto err; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	rc = pthread_create(&tid, NULL, fuse_loop_new_thread, bfuse); | 
					
						
							|  |  |  | 	if (rc != 0) { | 
					
						
							|  |  |  | 		SPDK_ERRLOG("could not create thread: %s\n", spdk_strerror(rc)); | 
					
						
							|  |  |  | 		rc = -rc; | 
					
						
							|  |  |  | 		goto err; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	bfuse->fuse_tid = tid; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	rc = pthread_detach(tid); | 
					
						
							|  |  |  | 	if (rc != 0) { | 
					
						
							|  |  |  | 		SPDK_ERRLOG("could not detach thread for fuse loop thread: %s\n", spdk_strerror(rc)); | 
					
						
							|  |  |  | 		rc = -rc; | 
					
						
							|  |  |  | 		goto err; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	*_bfuse = bfuse; | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | err: | 
					
						
							|  |  |  | 	blobfs_fuse_free(bfuse); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return rc; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void | 
					
						
							| 
									
										
										
										
											2020-05-10 18:49:31 +00:00
										 |  |  | blobfs_fuse_stop(struct spdk_blobfs_fuse *bfuse) | 
					
						
							| 
									
										
										
										
											2019-09-25 08:47:02 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-08-20 07:06:29 +00:00
										 |  |  | 	if (bfuse) { | 
					
						
							|  |  |  | 		fuse_session_exit(fuse_get_session(bfuse->fuse_handle)); | 
					
						
							|  |  |  | 		pthread_kill(bfuse->fuse_tid, SIGINT); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-09-25 08:47:02 +00:00
										 |  |  | } |