<<

To FUSE or Not to FUSE: Performance of User-Space File Systems Bharath Kumar Reddy Vangoor, Stony Brook University; Vasily Tarasov, IBM Research-Almaden; Erez Zadok, Stony Brook University https://www.usenix.org/conference/fast17/technical-sessions/presentation/vangoor

This paper is included in the Proceedings of the 15th USENIX Conference on File and Storage Technologies (FAST ’17). February 27–March 2, 2017 • Santa Clara, CA, USA ISBN 978-1-931971-36-2

Open access to the Proceedings of the 15th USENIX Conference on File and Storage Technologies is sponsored by USENIX. To FUSE or Not to FUSE: Performance of User-Space File Systems Bharath Kumar Reddy Vangoor1, Vasily Tarasov2, and Erez Zadok1 1Stony Brook University and 2IBM Research—Almaden

Abstract whole systems, whereas user-space bugs’ impact is more contained. Many libraries and programming languages Traditionally, file systems were implemented as part are available in user-space in multiple platforms. of OS kernels. However, as complexity of file systems grew, many new file systems began being developed in Although user-space file systems are not expected . Nowadays, user-space file systems are often to displace kernel file systems entirely, they undoubt- used to prototype and evaluate new approaches to file edly occupy a growing niche, as some of the more system design. Low performance is considered the main heated debates between proponents and opponents in- disadvantage of user-space file systems but the dicate [20,39,41]. The debates center around two trade- of this problem has never been explored systematically. off factors: (1) how large is the performance overhead As a result, the topic of user-space file systems remains caused by a user-space implementations and (2) how rather controversial: while some consider user-space file much easier is it to develop in user space. Ease of de- systems a toy not to be used in production, others de- velopment is highly subjective, hard to formalize and velop full-fledged production file systems in user space. therefore evaluate; but performance is easier to evalu- In this paper we analyze the design and implementa- ate empirically. Oddly, little has been published on the tion of the most widely known user-space file system performance of user-space file system frameworks. framework—FUSE—and characterize its performance In this paper we use a popular user-space file system for a wide range of workloads. We instrumented FUSE framework, FUSE, and characterize its performance. We to extract useful statistics and traces, which helped us an- start with a detailed explanation of FUSE’s design and alyze its performance bottlenecks and present our anal- implementation for four reasons: (1) the architecture is ysis results. Our experiments indicate that depending on somewhat complex; (2) little information on internals is the workload and hardware used, performance degrada- available publicly; (3) FUSE’s source code can be dif- tion caused by FUSE can be completely imperceptible ficult to analyze, with complex asynchrony and user- or as high as –83% even when optimized; and relative kernel communications; and (4) as FUSE’s popularity CPU utilization can increase by 31%. grows, a detailed analysis of its implementation becomes of high value to many. 1 Introduction We developed a simple pass-through stackable file File systems offer a common interface for applications to system in FUSE and then evaluated its performance access data. Although micro-kernels implement file sys- when layered on top of compared to native Ext4. tems in user space [1, 16], most file systems are part of We used a wide variety of micro- and macro-workloads, monolithic kernels [6, 22, 34]. Kernel implementations and different hardware using basic and optimized config- avoid the high message-passing overheads of micro- urations of FUSE. We found that depending on the work- kernels and user-space daemons [7, 14]. load and hardware, FUSE can perform as well as Ext4, In recent years, however, user-space file systems but in the worst cases can be 3× slower. Next, we de- rose in popularity for four reasons. (1) Several stack- signed and built a rich instrumentation system for FUSE able file systems add specialized functionality over ex- to gather detailed performance metrics. The statistics ex- isting file systems (e.g., deduplication and compres- tracted are applicable to any FUSE-based systems. We sion [19, 31]). (2) In academia and R&D settings, this then used this instrumentation to identify bottlenecks in framework enabled quick experimentation and prototyp- FUSE, and to explain why, for example, its performance ing of new approaches [3, 9, 15, 21, 40]. (3) Several varied greatly for different workloads. existing kernel-level file systems were ported to user 2 FUSE Design space (e.g., ZFS [45], NTFS [25]). (4) More companies rely on user-space implementations: IBM’S GPFS [30] FUSE——is the most widely and LTFS [26], Nimble Storage’s CASL [24], Apache’s used user-space file system framework [35]. According HDFS [2], [13], RedHat’s Glus- to the most modest estimates, at least 100 FUSE-based terFS [29], Data Domain’s DDFS [46], etc. file systems are readily available on the Web [36]. Al- Increased file systems complexity is a contributing though other, specialized implementations of user-space factor to user-space file systems’ growing popularity file systems exist [30,32,42], we selected FUSE for this (e.g., is over 85 KLoC). User space code is eas- study because of its high popularity. ier to develop, port, and maintain. Kernel bugs can crash Although many file systems were implemented using

USENIX Association 15th USENIX Conference on File and Storage Technologies 59 FUSE—thanks mainly to the simple API it provides— tem, the daemon reads or writes from the device. little work was done on understanding its internal ar- When done with processing the request, the FUSE dae- chitecture, implementation, and performance [27]. For mon writes the response back to /dev/fuse; FUSE’s our evaluation it was essential to understand not only kernel driver then marks the request as completed and FUSE’s high-level design but also some details of its im- wakes up the original user process. plementation. In this section we first describe FUSE’s Some file system operations invoked by an application basics and then we explain certain important implemen- can complete without communicating with the user-level tation details. FUSE is available for several OSes: we FUSE daemon. For example, reads from a file whose selected due to its wide-spread use. We analyzed pages are cached in the kernel page cache, do not need the code of and ran experiments on the latest stable ver- to be forwarded to the FUSE driver. sion of the available at the beginning of the project—v4.1.13. We also used FUSE commit 2.2 Implementation Details #386b1b; on top of FUSE v2.9.4, this commit contains We now discuss several important FUSE implemen- several important patches which we did not want exclude tation details: the user-kernel protocol, library and from our evaluation. We manually examined all new API levels, in-kernel FUSE queues, splicing, multi- commits up to the time of this writing and confirmed threading, and write-back cache. that no new major features or improvements were added to FUSE since the release of the selected versions. Group (#) Request Types Special (3) INIT, DESTROY, INTERRUPT 2.1 High-Level Architecture Metadata (14) LOOKUP, FORGET, BATCH FORGET, CREATE, UNLINK, LINK, RENAME, RE- FUSE file−system daemon NAME2, OPEN, RELEASE, STATFS, Application FSYNC FLUSH ACCESS FUSE library , , User Data (2) READ, WRITE Attributes (2) GETATTR, SETATTR cache /dev/fuse VFS Other kernel Extended SETXATTR, GETXATTR, Q FUSE u subsystems Attributes (4) LISTXATTR, REMOVEXATTR e Kernel Kernel−based driver u Symlinks (2) SYMLINK, READLINK file system e (7) MKDIR, RMDIR, OPENDIR, RE- Figure 1: FUSE high-level architecture. LEASEDIR, READDIR, READDIRPLUS, FUSE consists of a kernel part and a user-level dae- FSYNCDIR Locking (3) GETLK, SETLK, SETLKW mon. The kernel part is implemented as a Linux kernel Misc (6) BMAP, FALLOCATE, MKNOD, IOCTL, module that, when loaded, registers a fuse file-system POLL, NOTIFY REPLY driver with Linux’s VFS. This Fuse driver acts as a proxy for various specific file systems implemented by differ- Table 1: FUSE request types, by group (whose size is in paren- ent user-level daemons. thesis). Requests we discuss in the text are in bold. In addition to registering a new file system, FUSE’s User-kernel protocol. When FUSE’s kernel driver kernel module also registers a /dev/fuse block de- communicates to the user-space daemon, it forms a vice. This device serves as an interface between user- FUSE request structure. Requests have different types space FUSE daemons and the kernel. In general, dae- depending on the operation they convey. Table 1 lists all mon reads FUSE requests from /dev/fuse, processes 43 FUSE request types, grouped by their semantics. As them, and then writes replies back to /dev/fuse. seen, most requests have a direct mapping to traditional Figure 1 shows FUSE’s high-level architecture. When VFS operations: we omit discussion of obvious requests a user application performs some operation on a (e.g., READ, CREATE) and instead next focus on those mounted FUSE file system, the VFS routes the operation less intuitive request types (marked bold in Table 1). to FUSE’s kernel driver. The driver allocates a FUSE The INIT request is produced by the kernel when a request structure and puts it in a FUSE queue. At this file system is mounted. At this point user space and point, the process that submitted the operation is usu- kernel negotiate (1) the protocol version they will op- ally put in a wait state. FUSE’s user-level daemon then erate on (7.23 at the time of this writing), (2) the set of picks the request from the kernel queue by reading from mutually supported capabilities (e.g., READDIRPLUS or /dev/fuse and processes the request. Processing the FLOCK support), and (3) various parameter settings (e.g., request might require re-entering the kernel again: for FUSE read-ahead size, time granularity). Conversely, example, in case of a stackable FUSE file system, the the DESTROY request is sent by the kernel during the daemon submits operations to the underlying file system file system’s unmounting process. When getting a DE- (e.g., Ext4); or in case of a block-based FUSE file sys- STROY, the daemon is expected to perform all necessary

60 15th USENIX Conference on File and Storage Technologies USENIX Association cleanups. No more requests will come from the kernel User FUSE Daemon for this session and subsequent reads from /dev/fuse Process

will return 0, causing the daemon to exit gracefully. sync sync, interrupts The INTERRUPT request is emitted by the kernel if async KERNEL USER any previously sent requests are no longer needed (e.g., T T T T T

when a user process blocked on a READ is terminated). reply Each request has a unique sequence# which INTERRUPT Fuse uses to identify victim requests. Sequence numbers are Async Kernel Page Driver assigned by the kernel and are also used to locate com- Cache H H H H pleted requests when the user space replies. Background Pending Processing Interrupts Forgets Every request also contains a node ID—an unsigned Dentry/ Interrupts Cache 64-bit integer identifying the inode both in kernel and Forgets Forgets user spaces. The path-to-inode translation is performed Figure 2: The organization of FUSE queues marked with their by the LOOKUP request. Every time an existing inode Head and Tail. The processing queue does not have a tail is looked up (or a new one is created), the kernel keeps because the daemon replies in an arbitrary order. the inode in the inode cache. When removing an in- space. This part exports the low-level FUSE API. ode from the dcache, the kernel passes the FORGET re- The High-level FUSE API builds on top of the low- quest to the user-space daemon. At this point the dae- level API and allows developers to skip the implemen- mon might decide to deallocate any corresponding data tation of the path-to-inode mapping. Therefore, neither structures. BATCH FORGET allows kernel to forget mul- nor lookup operations exist in the high-level API, tiple inodes with a single request. easing the code development. Instead, all high-level API An OPEN request is generated, not surprisingly, when methods operate directly on file paths. The high-level a user application opens a file. When replying to this re- API also handles request interrupts and provides other quest, a FUSE daemon has a chance to optionally assign convenient features: e.g., developers can use the more a 64-bit file handle to the opened file. This file handle common chown(), (), and truncate() is then returned by the kernel along with every request methods, instead of the lower-level setattr(). File associated with the opened file. The user-space daemon system developers must decide which API to use, by bal- can use the handle to store per-opened-file information. ancing flexibility vs. development ease. E.g., a stackable file system can store the descriptor of Queues. In Section 2.1 we mentioned that FUSE’s the file opened in the underlying file system as part of kernel has a request queue. FUSE actually maintains FUSE’s file handle. FLUSH is generated every time an five queues as seen in Figure 2: (1) interrupts, (2) for- opened file is closed; and RELEASE is sent when there gets, (3) pending, (4) processing, and (5) background.A are no more references to a previously opened file. request belongs to only one queue at any time. FUSE OPENDIR and RELEASEDIR requests have the same puts INTERRUPT requests in the interrupts queue, FOR- semantics as OPEN and RELEASE, respectively, but for GET requests in the forgets queue, and synchronous re- directories. The READDIRPLUS request returns one or quests (e.g., metadata) in the pending queue. When a more directory entries like READDIR, but it also includes file-system daemon reads from /dev/fuse, requests metadata information for each entry. This allows the ker- are transferred to the user daemon as follows: (1) Pri- nel to pre-fill its inode cache (similar to NFSv3’s READ- ority is given to requests in the interrupts queue; they DIRPLUS procedure [4]). are transferred to the user space before any other re- When the kernel evaluates if a user process has per- quest. (2) FORGET and non-FORGET requests are se- missions to access a file, it generates an ACCESS request. lected fairly: for each 8 non-FORGET requests, 16 FOR- By handling this request, the FUSE daemon can imple- GET requests are transferred. This reduces the bursti- ment custom permission logic. However, typically users ness of FORGET requests, while allowing other requests mount FUSE with the default permissions option that al- to proceed. The oldest request in the pending queue is lows kernel to grant or deny access to a file based on transferred to the user space and simultaneously moved its standard attributes (ownership and permission to the processing queue. Thus, processing queue re- bits). In this case no ACCESS requests are generated. quests are currently processed by the daemon. If the Library and API levels. Conceptually, the FUSE li- pending queue is empty then the FUSE daemon is brary consists of two levels. The lower level takes care blocked on the read call. When the daemon replies to of (1) receiving and parsing requests from the kernel, a request (by writing to /dev/fuse), the correspond- (2) sending properly formatted replies, (3) facilitating ing request is removed from the processing queue. file system configuration and mounting, and (4) hiding The background queue is for staging asynchronous potential version differences between kernel and user requests. In a typical setup, only read requests go to

USENIX Association 15th USENIX Conference on File and Storage Technologies 61 the background queue; writes go to the background explicit upper limit on the number of threads created queue too but only if the writeback cache is enabled. In by the FUSE library. An implicit limit exists for two such configurations, writes from the user processes are reasons: (1) by default, only 12 asynchronous requests first accumulated in the page cache and later bdflush (max background parameter) can be in the pend- threads wake up to flush dirty pages [8]. While flush- ing queue at one time; and (2) the number of syn- ing the pages FUSE forms asynchronous write requests chronous requests in the pending queue depends on the and puts them in the background queue. Requests from total amount of I/O activity generated by user processes. the background queue gradually trickle to the pending In addition, for every INTERRUPT and FORGET requests, queue. FUSE limits the number of asynchronous re- a new thread is invoked. In a typical system where there quests simultaneously residing in the pending queue to is no interrupts support and few FORGETs are generated, the configurable max background parameter (12 by the total number of FUSE daemon threads is at most default). When fewer than 12 asynchronous requests (12 + number of requests in pending queue). are in the pending queue, requests from the background Write back cache and max writes. The basic write queue are moved to the pending queue. The intention is behavior of FUSE is synchronous and only 4KB of data to limit the delay caused to important synchronous re- is sent to the user daemon for writing. This results in per- quests by bursts of background requests. formance problems on certain workloads; when copy- The queues’ lengths are not explicitly limited. How- ing a large file into a FUSE file system, /bin/cp indi- ever, when the number of asynchronous requests in the rectly causes every 4KB of data to be sent to userspace pending and processing queues reaches the value of the synchronously. The solution FUSE implemented was to congestion threshold tunable parameter (75% make FUSE’s page cache support a write-back policy max background of , 9 by default), FUSE informs the and then make writes asynchronous. With that change, Linux VFS that it is congested; the VFS then throttles file data can be pushed to the user daemon in larger the user processes that write to this file system. chunks of max write size (limited to 32 pages). Splicing and FUSE buffers. In its basic setup, the FUSE daemon has to read() requests from and 3 Instrumentation write() replies to /dev/fuse. Every such call To study FUSE’s performance, we developed a simple requires a memory copy between the kernel and user stackable passthrough file system—called Stackfs—and space. It is especially harmful for WRITE requests and instrumented FUSE’s kernel module and user-space li- READ replies because they often process a lot of data. brary to collect useful statistics and traces. We believe To alleviate this problem, FUSE can use splicing func- that the instrumentation presented here is useful for any- tionality provided by the Linux kernel [38]. Splicing one who develops a FUSE-based file system. allows the user space to transfer data between two in- kernel memory buffers without copying the data to user 3.1 Stackfs space. This is useful, e.g., for stackable file systems that pass data directly to the underlying file system. Stackfs is a file system that passes FUSE requests un- To seamlessly support splicing, FUSE represents its modified directly to the underlying file system. The rea- buffers in one of two forms: (1) a regular memory re- son for Stackfs was twofold. (1) After examining the gion identified by a pointer in the user daemon’s ad- code of all publicly available [28, 43] FUSE-based file dress space, or (2) a kernel-space memory pointed by systems, we found that most of them are stackable (i.e., a file descriptor. If a user-space file system implements deployed on top of other, often in-kernel file systems). the write buf() method, then FUSE splices the data (2) We wanted to add as little overhead as possible, to from /dev/fuse and passes the data directly to this isolate the overhead of FUSE’s kernel and library. method in a form of the buffer containing a file descrip- Complex production file systems often need a high de- tor. FUSE splices WRITE requests that contain more than gree of flexibility, and thus use FUSE’s low-level API. a single page of data. Similar logic applies to replies to As such file systems are our primary focus, we imple- READ requests with more than two pages of data. mented Stackfs using FUSE’s low-level API. This also Multithreading. FUSE added multithreading support avoided the overheads added by the high-level API. Be- as parallelism got more popular. In multi-threaded low we describe several important data structures and mode, FUSE’s daemon starts with one thread. If there procedures that Stackfs uses. are two or more requests available in the pending queue, Inode. Stackfs stores per-file metadata in an inode. FUSE automatically spawns additional threads. Every Stackfs’s inode is not persistent and exists in memory thread processes one request at a time. After process- only while the file system is mounted. Apart from book- ing the request, each thread checks if there are more keeping information, the inode stores the path to the un- than 10 threads; if so, that thread exits. There is no derlying file, its inode number, and a reference counter.

62 15th USENIX Conference on File and Storage Technologies USENIX Association The path is used, e.g., to open the underlying file when behavior. This file system is usually mounted at an OPEN request for a Stackfs file arrives. /sys/fs/fuse/connections/ and creates a di- Lookup. During lookup, Stackfs uses stat(2) to rectory for every mounted FUSE instance. Every direc- check if the underlying file exists. Every time a file is tory contains control files to abort a connection, check found, Stackfs allocates a new inode and returns the re- the total number of requests being processed, and adjust quired information to the kernel. Stackfs assigns its in- the upper limit and the threshold on the number of back- ode the number equal to the address of the inode struc- ground requests (see Section 2.2). We added 3 new files ture in memory (by typecasting), which is guaranteed to to these directories to export statistics from the in-kernel be unique. This allows Stackfs to quickly find the inode arrays. To export user-level array we added SIGUSR1 structure for any operations following the lookup (e.g., signal handler to the daemon. When triggered, the han- open or stat). The same inode can be looked up sev- dler prints the array to a log file specified during the dae- eral times (e.g., due to hardlinks) and therefore Stackfs mon’s start. The statistics captured have no measurable stores inodes in a hash table indexed by the underlying overhead on FUSE’s performance and are the primary inode number. When handling LOOKUP, Stackfs checks source of information about FUSE’s performance. the hash table to see whether the inode was previously Tracing. To understand FUSE’s behavior in more de- allocated and, if found, increases its reference counter tail we sometimes needed more information and had to by one. When a FORGET request arrives for an inode, resort to tracing. FUSE’s library already performs trac- Stackfs decreases inode’s reference count and deallo- ing when the daemon runs in debug mode but there is cates the inode when the count drops to zero. no tracing support for FUSE’s kernel module. We used File create and open. During file creation, Stackfs Linux’s static tracepoint mechanism [10] to add over 30 adds a new inode to the hash table after the correspond- tracepoints mainly to monitor the formation of requests ing file was successfully created in the underlying file during the complex writeback logic, reads, and some system. While processing OPEN requests, Stackfs saves metadata operations. Tracing helped us learn how fast the file descriptor of the underlying file in the file han- queues grow during our experiments, how much data is dle. The file descriptor is then used during read and write put into a single request, and why. operations and deallocated when the file is closed. Both FUSE’s statistics and tracing can be used by any existing and future FUSE-based file systems. The in- 3.2 Performance Statistics and Traces strumentation is completely transparent and requires no The existing FUSE instrumentation was insufficient for changes to file-system-specific code. in-depth FUSE performance analysis. We therefore in- 4 Methodology strumented FUSE to export important runtime statistics. Specifically, we were interested in recording the dura- FUSE has evolved significantly over the years and added tion of time that FUSE spends in various stages of re- several useful optimizations: writeback cache, zero- quest processing, both in kernel and user space. copy via splicing, and multi-threading. In our per- We introduced a two-dimensional array where a row sonal experience, some in the storage community tend to index (0–42) represents the request type and the column pre-judge FUSE’s performance—assuming it is poor— index (0–31) represents the time. Every cell in the array mainly due to not having information about the improve- stores the number of requests of a corresponding type ments FUSE made over the years. We therefore de- that were processed within the 2N+1–2N+2 nanosec- signed our methodology to evaluate and demonstrate onds where N is the column index. The time dimension how FUSE’s performance advanced from its basic con- therefore covers the interval of up to 8 seconds which figurations to ones that include all of the latest optimiza- is enough in typical FUSE setups. (This technique effi- tions. We now detail our methodology, starting from the ciently records a log2 latency histogram [18].) We then description of FUSE configurations, proceed to the list added four such arrays to FUSE: the first three arrays of workloads, and finally present our testbed. are in the kernel, capturing the time spent by the re- FUSE configurations. To demonstrate the evolution quest inside the background, pending, and processing of FUSE’s performance, we picked two configurations queues. For the processing queue, the captured time also on opposite ends of the spectrum: the basic configu- includes the time spent by requests in user space. The ration (called StackfsBase) with no major FUSE opti- fourth array is in user space and tracks the time the dae- mizations and the optimized configuration (called Stack- mon needs to process a request. The total memory size fsOpt) that enables all FUSE improvements available of all four arrays is only 48KiB and only few instructions as of this writing. Compared to StackfsBase, the are necessary to update values in the array. StackfsOpt configuration adds the following features: FUSE includes a special fusectl file system (1) writeback cache is turned on; (2) maximum size to allow users to control several aspects of FUSE’s of a single FUSE request is increased from 4KiB to

USENIX Association 15th USENIX Conference on File and Storage Technologies 63 128KiB (max write parameter); (3) user daemon runs our results more effectively, we split the evaluation in in the multi-threaded mode; (4) splicing is activated for two. Section 5.1 overviews our extensive evaluation all operations (splice read, splice write, and results—most useful information for many practitioners. splice move parameters). We left all other parame- Detailed performance analysis follows in Section 5.2. ters at their default values in both configurations. Workloads. To stress different modes of FUSE op- 5.1 Performance Overview eration and conduct an thorough performance charac- To evaluate FUSE’s performance degradation, we first terization, we selected a broad set of workloads: mi- measured the throughput (in ops/sec) achieved by native cro and macro, metadata- and data-intensive, and also Ext4 and then measured the same for Stackfs deployed experimented with a wide range of I/O sizes and par- over Ext4. As detailed in Section 4 we used two config- allelism levels. Table 2 describes all workloads that urations of Stackfs: a basic (StackfsBase) and optimized we employed. To simplify the identification of work- (StackfsOpt) one. From here on, we use Stackfs to re- loads in the text we use the following mnemonics: rnd fer to both of these configurations. We then calculated stands for random, seq for sequential, rd for reads, the relative performance degradation (or improvement) wr for writes, cr for creates, and del for deletes. of Stackfs vs. Ext4 for each workload. Table 3 shows The presence of Nth and Mf substrings in a workload absolute throughputs for Ext4 and relative performance name means that the workload contains N threads and M for two Stackfs configurations for both HDD and SSD. files, respectively. We fixed the amount of work (e.g., For better clarity we categorized the results by the number of reads in rd workloads) rather than the Stackfs’s performance difference into four classes: amount of time in every experiment. We find it eas- (1) The Green class (marked with +) indicates that the ier to analyze performance in experiments with a fixed performance either degraded by less than 5% or actu- amount of work. We picked a sufficient amount of ally improved; (2) The Yellow class (*) includes results work so that the performance stabilized. Resulting run- with the performance degradation in the 5–25% range; times varied between 8 and 20 minutes across the exper- (3) The Orange class (#) indicates that the performance iments. Because SSDs are orders of magnitude faster degradation is between 25–50%; And finally, (4) the Red than HDDs, for some workloads we selected a larger class (!) is for when performance decreased by more than amount of work for our SSD-based experiments. We 50%. Although the ranges for acceptable performance used Filebench [12, 37] to generate all workloads. degradation depend on the specific deployment and the Experimental setup. FUSE performance depends value of other benefits provided by FUSE, our classifi- heavily on the speed of the underlying storage: faster de- cation gives a broad overview of FUSE’s performance. vices expose FUSE’s own overheads. We therefore ex- Below we list our main observations that characterize perimented with two common storage devices of differ- the results. We start from the general trends and move to ent speed: an HDD (Seagate Savvio 15K.2, 15KRPM, more specific results towards the end of the list. 146GB) and an SSD ( X25-M SSD, 200GB). Both devices were installed in three identical Dell PowerEdge Observation 1. The relative difference varied across R710 machines with 4-core Intel Xeon E5530 2.40GHz workloads, devices, and FUSE configurations from -cr-1th CPU each. The amount of RAM available to the OS was –83.1% for [row #37] to +6.2% for web-server set to 4GB to accelerate cache warmup in our experi- [row #45]. ments. The machines ran CentOS 7 with Linux kernel Observation 2. For many workloads, FUSE’s opti- upgraded to v4.1.13 and FUSE library commit #386b1b. mizations improve performance significantly. E.g., for We used Ext4 [11] as the underlying file system be- the web-server workload, StackfsOpt improves per- cause it is common, stable, and has a well documented formance by 6.2% while StackfsBase degrades it by design which facilitates performance analysis. Before more than 50% [row #45]. every experiment we reformatted the storage devices Observation 3. Although optimizations increase the with Ext4 and remounted the file systems. To lower performance of some workloads, they can degrade the the variability in our experiments we disabled Ext4’s performance of others. E.g., StackfsOpt decreases lazy inode initialization [5]. In either case, standard de- performance by 35% more than StackfsBase for the viations in our experiments were less than 2% for all files-rd-1th workload on SSD [row #39]. workloads except for three: seq-rd-1th-1f (6%), files-rd-32th (7%), and mail-server (7%). Observation 4. In the best performing configuration of Stackfs (among StackfsOpt and StackfsBase) only 5 Evaluation two file-create workloads (out of a total 45 workloads) For many, FUSE is just a practical tool to build real prod- fell into the red class: files-cr-1th [row #37] and ucts or prototypes, but not a research focus. To present files-cr-32th [row #38].

64 15th USENIX Conference on File and Storage Technologies USENIX Association Workload Name Description Results seq-rd-Nth-1f N threads (1, 32) sequentially read from a single preallocated 60GB file. [rows #1–8] seq-rd-32th-32f 32 threads sequentially read 32 preallocated 2GB files. Each thread reads its own file. [rows #9–12] rnd-rd-Nth-1f N threads (1, 32) randomly read from a single preallocated 60GB file. [rows #13–20] seq-wr-1th-1f Single thread creates and sequentially writes a new 60GB file. [rows #21–24] seq-wr-32th-32f 32 threads sequentially write 32 new 2GB files. Each thread writes its own file. [rows #25–28] rnd-wr-Nth-1f N threads (1, 32) randomly write to a single preallocated 60GB file. [rows #29–36] files-cr-Nth N threads (1, 32) create 4 million 4KB files over many directories. [rows #37–38] files-rd-Nth N threads (1, 32) read from 1 million preallocated 4KB files over many directories. [rows #39–40] files-del-Nth N threads (1, 32) delete 4 million of preallocated 4KB files over many directories. [rows #41–42] file-server File-server workload emulated by Filebench. Scaled up to 200,000 files. [row #43] mail-server Mail-server workload emulated by Filebench. Scaled up to 1.5 million files. [row #44] web-server Web-server workload emulated by Filebench. Scaled up to 1.25 million files. [row #45] Table 2: Description of workloads and their corresponding result rows. For data-intensive workloads, we experimented with 4KB, 32KB, 128KB, and 1MB I/O sizes. We picked dataset sizes so that both cached and non-cached data are exercised. The Results column correlates these descriptions with results in Table 3. Observation 5. Stackfs’s performance depends signif- 5.2.1 Read Workloads icantly on the underlying device. E.g., for sequential background queue read workloads [rows #1–12], Stackfs shows no per- 1,048,576 451K pending queue formance degradation for SSD and a 26–42% degrada- processing queue ) user daemon 2 65536 tion for HDD. The situation is reversed, e.g., when a mail-server [row #44] workload is used. 4096

256 Observation 6. At least in one Stackfs configuration, 34 32 32 32 Requests (log all write workloads (sequential and random) [rows #21– Number of FUSE 16 1 1 36] are within the Green class for both HDD and SSD. 0 INIT LOOKUP OPEN READ FLUSH RELEASE GETATTR Observation 7. The performance of sequential read [rows #1–12] are well within the Green class for both Types of FUSE Requests HDD and SSD; however, for the seq-rd-32th-32f Figure 3: Different types and number of requests generated by workload [rows #5–8] on HDD, they are in Orange StackfsBase on SSD during the seq-rd-32th-32f work- class. Random read workload results [rows #13–20] load, from left to right, in their order of generation. span all four classes. Furthermore, the performance Figure 3 demonstrates the types of requests that were grows as I/O sizes increase for both HDD and SSD. generated with the seq-rd-32th-32f workload. We Observation 8. In general, Stackfs performs visi- use seq-rd-32th-32f as a reference for the figure bly worse for metadata-intensive and macro workloads because this workload has more requests per operation [rows #37–45] than for data-intensive workloads [rows type compared to other workloads. Bars are ordered #1–36]. The performance is especially low for SSDs. from left to right by the appearance of requests in the ex- periment. The same request types, but in different quan- Observation 9. The relative CPU utilization of Stackfs tities, were generated by the other read-intensive work- (not shown in the Table) is higher than that of Ext4 and loads [rows #1–20]. For the single threaded read work- varies in the range of +0.13% to +31.2%; similarly, CPU loads, only one request per LOOKUP, OPEN, FLUSH, and cycles per operation increased by 1.2× to 10× times be- RELEASE type was generated. The number of READ tween Ext4 and Stackfs (in both configurations). This requests depended on the I/O size and the amount of behavior is seen in both HDD and SSD. data read; INIT request is produced at mount time so its Observation 10. CPU cycles per operation are count remained the same across all workloads; and fi- higher for StackfsOpt than for StackfsBase for nally GETATTR is invoked before unmount for the root the majority of workloads. But for the work- directory and was the same for all the workloads. loads seq-wr-32th-32f [rows #25–28] and Figure 3 also shows the breakdown of requests by rnd-wr-1th-1f [rows #30–32], StackfsOpt con- queues. By default, READ, RELEASE, and INIT are asyn- sumes fewer CPU cycles per operation. chronous; they are added to the background queue first. 5.2 Analysis All other requests are synchronous and are added to pending queue directly. In read workloads, only READ We analyzed FUSE performance results and present requests are generated in large numbers. Thus, we dis- main findings here, following the order in Table 3. cuss in detail only READ requests for these workloads.

USENIX Association 15th USENIX Conference on File and Storage Technologies 65 I/O Size HDD Results SSD Results # Workload (KB) EXT4 StackfsBase StackfsOpt EXT4 StackfsBase StackfsOpt (ops/sec) (%Diff) (%Diff) (ops/sec) (%Diff) (%Diff) 1 4 38382 - 2.45+ + 1.7+ 30694 - 0.5+ - 0.9+ seq-rd- 2 32 4805 - 0.2+ - 2.2+ 3811 + 0.8+ + 0.3+ 1th-1f 3 128 1199 - 0.86+ - 2.1+ 950 + 0.4+ + 1.7+ 4 1024 150 - 0.9+ - 2.2+ 119 + 0.2+ - 0.3+ 5 4 1228400 - 2.4+ - 3.0+ 973450 + 0.02+ + 2.1+ seq-rd- 6 32 153480 - 2.4+ - 4.1+ 121410 + 0.7+ + 2.2+ 32th-1f 7 128 38443 - 2.6+ - 4.4+ 30338 + 1.5+ + 1.97+ 8 1024 4805 - 2.5+ - 4.0+ 3814.50 - 0.1+ - 0.4+ 9 4 11141 - 36.9# - 26.9# 32855 - 0.1+ - 0.16+ seq-rd- 10 32 1491 - 41.5# - 30.3# 4202 - 0.1+ - 1.8+ 32th-32f 11 128 371 - 41.3# - 29.8# 1051 - 0.1+ - 0.2+ 12 1024 46 - 41.0# - 28.3# 131 - 0.03+ - 2.1+ 13 4 243 - 9.96* - 9.95* 4712 - 32.1# - 39.8# rnd-rd- 14 32 232 - 7.4* - 7.5* 2032 - 18.8* - 25.2# 1th-1f 15 128 191 - 7.4* - 5.5* 852 - 14.7* - 12.4* 16 1024 88 - 9.0* -3.1+ 114 - 15.3* -1.5+ 17 4 572 - 60.4! -23.2* 24998 - 82.5! -27.6# rnd-rd- 18 32 504 - 56.2! -17.2* 4273 - 55.7! -1.9+ 32th-1f 19 128 278 - 34.4# -11.4* 1123 - 29.1# -2.6+ 20 1024 41 - 37.0# -15.0* 126 - 12.2* -1.9+ 21 4 36919 -26.2# - 0.1+ 32959 - 9.0* + 0.1+ seq-wr- 22 32 4615 - 17.8* - 0.16+ 4119 - 2.5+ + 0.12+ 1th-1f 23 128 1153 - 16.6* - 0.15+ 1030 - 2.1+ + 0.1+ 24 1024 144 - 17.7* -0.31+ 129 - 2.3+ - 0.08+ 25 4 34370 - 2.5+ + 0.1+ 32921 + 0.05+ + 0.2+ seq-wr- 26 32 4296 - 2.7+ + 0.0+ 4115 + 0.1+ + 0.1+ 32th-32f 27 128 1075 - 2.6+ - 0.02+ 1029 - 0.04+ + 0.2+ 28 1024 134 - 2.4+ - 0.18+ 129 - 0.1+ + 0.2+ 29 4 1074 - 0.7+ - 1.3+ 16066 + 0.9+ - 27.0# rnd-wr- 30 32 708 - 0.1+ - 1.3+ 4102 - 2.2+ - 13.0* 1th-1f 31 128 359 - 0.1+ - 1.3+ 1045 - 1.7+ - 0.7+ 32 1024 79 - 0.01+ - 0.8+ 129 - 0.02+ - 0.3+ 33 4 1073 - 0.9+ - 1.8+ 16213 - 0.7+ - 26.6# rnd-wr- 34 32 705 + 0.1+ - 0.7+ 4103 - 2.2+ - 13.0* 32th-1f 35 128 358 + 0.3+ - 1.1+ 1031 - 0.1+ + 0.03+ 36 1024 79 + 0.1+ - 0.3+ 128 + 0.9+ - 0.3+ 37 files-cr-1th 4 30211 - 57! - 81.0! 35361 - 62.2! - 83.3! 38 files-cr-32th 4 36590 - 50.2! - 54.9! 46688 - 57.6! - 62.6! 39 files-rd-1th 4 645 + 0.0+ - 10.6* 8055 - 25.0* - 60.3! 40 files-rd-32th 4 1263 - 50.5! -4.5+ 25341 - 74.1! -33.0# 41 files-del-1th - 1105 - 4.0+ - 10.2* 7391 - 31.6# - 60.7! 42 files-del-32th - 1109 - 2.8+ - 6.9* 8563 - 42.9# - 52.6! 43 file-server - 1705 - 26.3# -1.4+ 5201 - 41.2# -1.5+ 44 mail-server - 1547 - 45.0# -4.6+ 11806 - 70.5! -32.5# 45 web-server - 1704 - 51.8! +6.2+ 19437 - 72.9! -17.3* Table 3: List of workloads and corresponding performance results. Green class (marked with +) indicates that the performance either degraded by less than 5% or actually improved; Yellow class (*) includes results with the performance degradation in the 5–25% range; Orange class (#) indicates that the performance degradation is between 25–50%; And finally, the Red class (!) is for when performance decreased by more than 50%.

66 15th USENIX Conference on File and Storage Technologies USENIX Association Sequential Read using 1 thread on 1 file. The total The smaller the I/O size is, the more READ requests number of READ requests that StackfsBase generated are generated and the higher FUSE’s overhead tended during the whole experiment for different I/O sizes for to be. This is seen for StackfsOpt where performance HDD and SSD remained approximately the same and for HDD gradually grows from –10.0% for 4KB to –3% equal to 491K. Our analysis revealed that this happens for 1MB I/O sizes. A similar situation is seen for SSD. because of FUSE’s default 128KB-size readahead which Thanks to splice, StackfsOpt performs better than Stack- effectively levels FUSE request sizes no matter what is fsBase for large I/O sizes. For 1MB I/O size, the im- the user application I/O size. Thanks to readahead, se- provement is 6% on HDD and 14% on SSD. Interest- quential read performance of StackfsBase and Stackf- ingly, 4KB I/O sizes have the highest overhead because sOpt was as good as Ext4 for both HDD and SSD. FUSE splices requests only if they are larger than 4KB. Sequential Read using 32 threads on 32 files. Due to Random Read using 32 threads on 1 file. Similar readahead, the total number of READ requests generated to the previous experiment (single thread random read), here was also approximately same for different I/O sizes. readahead does not help smaller I/O sizes here: every At any given time, 32 threads are requesting data and user read call is sent to the user daemon and causes continuously add requests to queues. StackfsBase and high performance degradation: up to –83% for Stackfs- StackfsOpt show significantly larger performance degra- Base and –28% for StackfsOpt. The overhead caused by dation on HDD compared to SSD. For StackfsBase, the StackfsBase is high in these experiments (up to –60% for user daemon is single threaded and the device is slower, HDD and –83% for SSD), for both HDD and SSD, and so requests do not move quickly through the queues. On especially for smaller I/O sizes. This is because when 32 the faster SSD, however, even though the user daemon user threads submit a READ request, 31 of those threads is single threaded, requests move faster in the queues. need to wait while the single-threaded user daemon pro- Hence performance of StackfsBase is as close to that cesses one request at a time. StackfsOpt reduced perfor- of Ext4. With StackfsOpt, the user daemon is multi- mance degradation compared to StackfsBase, but not as threaded and can fill the HDD’s queue faster so perfor- much for 4KB I/Os because splice is not used for request mance improved for HDD compared to SSD. that are smaller or equal to 4KB. Investigating further, we found that for HDD and StackfsOpt, FUSE’s daemon was bound by the 5.2.2 Write Workloads max background value (default is 12): at most, only 12 user deamons (threads) were spawned. We increased background queue pending queue 15M that limit to 100 and reran the experiments: now Stackf- 16,777,216 processing queue user daemon ) sOpt was within 2% of Ext4’s performance. 2 1,048,576 Sequential Read using 32 threads on 1 file. 65,536 This workload exhibits similar performance trends to 4,096

seq-rd-1th-1f. However, because all 32 user Requests(log 256 Number of FUSE 34 32 32 32 12 threads read from the same file, they benefit from the 16 1 1 shared page cache. As a result, instead of 32× more 0 INIT LOOKUP GETATTR CREATE GETXATTR WRITE FLUSH RELEASE FUSE requests, we saw only up to a 37% increase in number of requests. This modest increase is because, Types of FUSE Requests in the beginning of the experiment, every thread tries to Figure 4: Different types of requests that were generated by read the data separately; but after a certain point in time, StackfsBase on SSD for the seq-wr-32th-32f workload, only a single thread’s requests are propagated to the user from left to right in their order of generation. daemon while all other threads’ requests are available We now discuss the behavior of StackfsBase and in the page cache. Also, having 32 user threads run- StackfsOpt in all write workloads listed in Table 3 ning left less CPU time available for FUSE’s threads to [rows #21–36]. Figure 4 shows the different types execute, thus causing a slight (up to 4.4%) decrease in of requests that got generated during all write work- performance compared to Ext4. loads, from left to right in their order of generation Random Read using 1 thread on 1 file. Unlike the (seq-wr-32th-32f is used as a reference). In case case of small sequential reads, small random reads did of rnd-wr workloads, CREATE requests are replaced not benefit from FUSE’s readahead. Thus, every appli- by OPEN requests, as random writes operate on pre- cation read call was forwarded to the user daemon which allocated files. For all the seq-wr workloads, due to resulted in an overhead of up to 10% for HDD and 40% the creation of files, a GETATTR request was generated to for SSD. The absolute Ext4 throughput is about 20× check permissions of the single directory where the files higher for SSD than for HDD which explains the higher were created. Linux VFS caches attributes and therefore penalty on FUSE’s relative performance on SSD. there were fewer than 32 GETATTRs. For single-threaded

USENIX Association 15th USENIX Conference on File and Storage Technologies 67 Stages of write() call processing Time Time Sequential Write using 32 threads on 32 files. Per- (µs) (%) formance trends are similar to seq-wr-1th-1f but Processing by VFS before passing execu- 1.4 2.4 even the unoptimized StackfsBase performed much bet- tion to FUSE kernel code ter (up to –2.7% and –0.1% degradation for HDD and FUSE request allocation and initialization 3.4 6.0 SSD, respectively). This is because without the write- Waiting in queues and copying to user space 10.7 18.9 back cache, 32 user threads put more requests into Processing by Stackfs daemon, includes 24.6 43.4 Ext4 execution FUSE’s queues (compared to 1 thread) and therefore Processing reply by FUSE kernel code 13.3 23.5 kept the user daemon constantly busy. Processing by VFS after FUSE kernel code 3.3 5.8 Random Write using 1 thread on 1 file. Performance Total 56.7 100.0 degradation caused by StackfsBase and StackfsOpt was low on HDD for all I/O sizes (max –1.3%) because the Table 4: Average latencies of a single write request generated random write performance of Ext4 on HDD is low— by StackfsBase during seq-wr-4KB-1th-1f workload across between 79 and 1,074 Filebench ops/sec, depending on multiple profile points on HDD. the I/O size (compare to over 16,000 ops/sec for SSD). workloads, five operations generated only one request: The performance bottleneck, therefore, was in the HDD LOOKUP, OPEN, CREATE, FLUSH, and RELEASE; how- I/O time and FUSE overhead was invisible. ever, the number of WRITE requests was orders of mag- Interestingly, on SSD, StackfsOpt performance degra- nitude higher and depended on the amount of data writ- dation was high (–27% for 4KB I/O) and more than the ten. Therefore, we consider only WRITE requests when StackfsBase for 4KB and 32KB I/O sizes. The reason we discuss each workload in detail. for this is that currently FUSE’s writeback cache batches Usually the Linux VFS generates GETXATTR before only sequential writes into a single WRITE. Therefore, every write operation. But in our case StackfsBase in the case of random writes there is no reduction in and StackfsOpt did not support extended attributes and the number of WRITE requests compared to StackfsBase. the kernel cached this knowledge after FUSE returned These numerous requests are processed asynchronously ENOSUPPORT for the first GETXATTR. (i.e., added to the background queue). And because of Sequential Write using 1 thread on 1 file. The to- FUSE’s congestion threshold on the background queue tal number of WRITE requests that StackfsBase gener- the application that is writing the data becomes throttled. ated during this experiment was 15.7M for all I/O sizes. For I/O size of 32KB, StackfsOpt can pack the entire This is because in StackfsBase each user write call is 32KB into a single WRITE request. Compared to Stack- split into several 4KB-size FUSE requests which are sent fsBase, this reduces the number of WRITE requests by to the user daemon. As a result StackfsBase degraded 8× and results in 15% better performance. performance ranged from –26% to –9%. Compared to Random Write using 32 threads on 1 file. This StackfsBase, StackfsOpt generated significantly fewer workload performs similarly to rnd-wr-1th-1f and FUSE requests: between 500K and 563K depending on the same analysis applies. the I/O size. The reason is the writeback cache that al- lows FUSE’s kernel part to pack several dirty pages (up 5.2.3 Metadata Workloads to 128KB in total) into a single WRITE request. Ap- We now discuss the behavior of Stackfs in all metadata 1 micro-workloads as listed in Table 3 [rows #37–42]. proximately 32 of requests were generated in StackfsOpt compared to StackfsBase. This suggests indeed that File creates. Different types of requests that got gen- each WRITE request transferred about 128KB of data (or erated during the files-cr-Nth runs are GETATTR, 32× more than 4KB). LOOKUP, CREATE, WRITE, FLUSH, RELEASE, and FOR- Table 4 shows the breakdown of time spent (latencies) GET. The total number of each request type gener- by a single write request across various stages, during ated was exactly 4 million. Many GETATTR requests the seq-wr-4KB-1th-1f workload on HDD. Taking were generated due to Filebench calling a fstat on only major latencies, the write request spends 19% of its the file to check whether it exists or not before creat- time in request creation and waiting in the kernel queues; ing it. Files-cr-Nth workloads demonstrated the 43% of its time in user space, which includes time taken worst performance among all workloads for both Stack- by the underlying Ext4 to serve the write; and then 23% fsBase and StackfsOpt and for both HDD and SSD. The of time during copy of the response from user space to reason is twofold. First, for every single file create, five kernel. The relative CPU utilization caused by Stack- operations happened serially: GETATTR, LOOKUP, CRE- fsBase and StackfsOpt in seq-wr-4KB-1th-1f on ATE, WRITE, and FLUSH; and as there were many files HDD is 6.8% and 11.1% more than native Ext4, re- accessed, they all could not be cached, so we saw many spectively; CPU cycles per operation were the same for FORGET requests to remove cached items—which added StackfsBase and StackfsOpt—4× that of native Ext4. further overhead. Second, file creates are fairly fast in

68 15th USENIX Conference on File and Storage Technologies USENIX Association 2.5e+06 background queue processing queue (7–8× higher throughput than the HDD). This neg- pending queue user daemon 2M 2e+06 atively impacted Stackfs in terms of relative num- bers: its performance degradation was 25–50% higher 1.5e+06 on SSD than on HDD. In all cases StackfsOpt’s per- 1M 1M 1M 1M 1M 1M 1e+06

Requests 780K formance degradation is more than StackfsBase’s be-

Number of FUSE 500000 cause neither splice nor the writeback cache helped files-del-Nth workloads and only added addi- 1 0 INIT LOOKUP OPEN GETATTR READ FLUSH RELEASE FORGET tional overhead for managing extra threads.

Types of FUSE Requests 5.2.4 Macro Server Workloads Figure 5: Different types of requests that were generated by We now discuss the behavior of Stackfs for macro- StackfsBase on SSD for the files-rd-1th workload, from workloads [rows #43–45]. left to right in their order of generation. background queue processing queue Ext4 (30–46 thousand creates/sec) because small newly 268,435,456 pending queue user daemon 16,777,216 8.7M ) created inodes can be effectively cached in RAM. Thus, 2 1.4M 1,048,576 364K 500K 427K 750K 750K

250K 250K 250K 250K overheads caused by the FUSE’s user-kernel communi- 65,536 cations explain the performance degradation. 4,096

Requests(log 256 File Reads. Figure 5 shows different types of requests Number of FUSE 16 files-rd-1th 2 that got generated during the work- 0 1 load. This workload is metadata-intensive because it INIT LOOKUPGETATTRCREATE OPEN READ GETXATTRWRITE FLUSH RELEASEUNLINK FORGET contains many small files (one million 4KB files) that are repeatedly opened and closed. Figure 5 shows that half Types of FUSE Requests of the READ requests went to the background queue and Figure 6: Different types of requests that were generated by the rest directly to the pending queue. The reason is that StackfsBase on SSD for the file-server workload. when reading a whole file, and the application requests File Server. Figure 6 shows different types of oper- reads beyond the EOF, FUSE generates a synchronous ations that got generated during the file-server READ request which goes to the pending queue (not the workload. Macro workloads are expected to have a background queue). Reads past the EOF also generate a more diverse request profile than micro workloads, and GETATTR request to confirm the file’s size. file-server confirms this: many different requests The performance degradation for files-rd-1th got generated, with WRITEs being the majority. in StackfsBase on HDD is negligible; on SSD, however, The performance improved by 25–40% (depending the relative degradation is high (–25%) because the SSD on storage device) with StackfsOpt compared to Stack- is 12.5× faster than HDD (see Ext4 absolute through- fsBase, and got close to Ext4’s native performance for put in Table 3). Interestingly, StackfsOpt’s performance three reasons: (1) with a writeback cache and 128KB re- degradation is more than that of StackfsBase (by 10% quests, the number of WRITEs decreased by a factor of and 35% for HDD and SSD, respectively). The rea- 17× for both HDD and SSD, (2) with splice, READ and son is that in StackfsOpt, different FUSE threads pro- WRITE requests took advantage of zero copy, and (3) the cess requests for the same file, which requires additional user daemon is multi-threaded, as the workload is. synchronization and context switches. Conversely, but 268,435,456 background queue processing queue as expected, for files-rd-32th workload, Stack- pending queue user daemon 16,777,216 ) fsOpt performed 40–45% better than StackfsBase be- 2 2M 1.4M 1.8M 2.5M 1.5M1.5M

1M 1M 1,048,576 0.5M 1M 0.5M 0.5M

cause multiple threads are needed to effectively process 65,536

parallel READ requests. 4,096

Requests(log 256

File Deletes. The different types of operations that got Number of FUSE generated during the files-del-1th workloads are 16 0 1 1 LOOKUP, UNLINK, FORGET (exactly 4 million each). INIT LOOKUPGETATTRCREATEOPEN READ GETXATTRWRITE FSYNC FLUSH RELEASEUNLINKFORGET Every UNLINK request is followed by FORGET. There- fore, for every incoming delete request that the appli- Types of FUSE Requests cation (Filebench) submits, StackfsBase and StackfsOpt Figure 7: Different types of requests that were generated by generates three requests (LOOKUP, UNLINK, and FOR- StackfsBase on SSD for the mail-server workload. GET) in series, which depend on each other. Mail Server. Figure 7 shows different types of op- Deletes translate to small random writes at the block erations that got generated during the mail-server layer and therefore Ext4 benefited from using an SSD workload. As with the file-server workload, many

USENIX Association 15th USENIX Conference on File and Storage Technologies 69 tion about FUSE design at the time. 268,435,456 background queue processing queue pending queue user daemon 16,777,216 Second, in a position paper, Tarasov et al. character- ) 5M

2 2.8M 2.7M 2.5M 2.5M 2.5M 2.6M 1.5M 1,048,576 0.8M ized FUSE performance for a variety of workloads but

65,536 did not analyze the results [36]. Furthermore, they evalu- 4,096 ated only default FUSE configuration and discussed only

Requests(log 256

Number of FUSE FUSE’s high-level architecture. In this paper we evalu- 16 ated and analyzed several FUSE configurations in detail, 0 1 1 INIT LOOKUP GETATTR OPEN READ GETXATTRWRITE FLUSH RELEASE FORGET and described FUSE’s low-level architecture. Several researchers designed and implemented use- Types of FUSE Requests ful extensions to FUSE. Re-FUSE automatically restarts Figure 8: Different types of requests that were generated by FUSE file systems that crash [33]. To improve FUSE StackfsBase on SSD for the web-server workload. performance, Narayan et al. proposed to marry in-kernel different requests got generated, with WRITEs being the stackable file systems [44] with FUSE [23]. Shun et al. majority. Performance trends are also similar between modified FUSE’s kernel module to allow applications these two workloads. However, in the SSD setup, even to access storage devices directly [17]. These improve- the optimized StackfsOpt still did not perform close to ments were in research prototypes and were never in- Ext4 in this mail-server workload, compared to cluded in the mainline. file-server. The reason is twofold. First, com- 7 Conclusion pared to file server, mail server has almost double the User-space file systems are popular for prototyping new metadata operations, which increases FUSE overhead. ideas and developing complex production file systems Second, I/O sizes are smaller in mail-server which im- that are difficult to maintain in kernel. Although many proves the underlying Ext4 SSD performance and there- researchers and companies rely on user-space file sys- fore shifts the bottleneck to FUSE. tems, little attention was given to understanding the per- Web Server. Figure 8 shows different types of re- formance implications of moving file systems to user quests generated during the web-server workload. space. In this paper we first presented the detailed design This workload is highly read-intensive as expected from of FUSE, the most popular user-space file system frame- a Web-server that services static Web-pages. The perfor- work. We then conducted a broad performance charac- mance degradation caused by StackfsBase falls into the terization of FUSE and we present an in-depth analysis Red class in both HDD and SSD. The major bottleneck of FUSE performance patterns. We found that for many was due to the FUSE daemon being single-threaded, workloads, an optimized FUSE can perform within 5% while the workload itself contained 100 user threads. of native Ext4. However, some workloads are unfriendly Performance improved with StackfsOpt significantly on to FUSE and even if optimized, FUSE degrades their both HDD and SSD, mainly thanks to using multiple performance by up to 83%. Also, in terms of the CPU threads. In fact, StackfsOpt performance on HDD is utilization, the relative increase seen is 31%. even 6% higher than of native Ext4. We believe this All of our code and Filebench workloads files are minor improvement is caused by the Linux VFS treating available from http://filesystems.org/fuse/. Stackfs and Ext4 as two independent file systems and Future work. There is a large room for improvement allowing them together to cache more data compared to in FUSE performance. We plan to add support for com- when Ext4 is used alone, without Stackfs. This does not pound FUSE requests and investigate the possibility of help SSD setup as much due to the high speed of SSD. shared memory between kernel and user spaces for faster 6 Related Work communications. Many researchers used FUSE to implement file sys- Acknowledgments tems [3, 9, 15, 40] but little attention was given to under- We thank the anonymous FAST reviewers and our shep- standing FUSE’s underlying design and performance. herd Tudor Marian for their valuable comments. This To the best of our knowledge, only two papers stud- work was made possible in part thanks to Dell-EMC, ied some aspects of FUSE. First, Rajgarhia and Gehani NetApp, and IBM support; NSF awards CNS-1251137, evaluated FUSE performance with Java bindings [27]. CNS-1302246, CNS-1305360, and CNS-1622832; and Compared to this work, they focused on evaluating Java ONR award 12055763. library wrappers, used only three workloads, and ran experiments with FUSE v2.8.0-pre1 (released in 2008). The version they used did not support zero-copying via splice, writeback caching, and other important fea- tures. The authors also presented only limited informa-

70 15th USENIX Conference on File and Storage Technologies USENIX Association References ACM Symposium on Operating Systems Principles [1] M. Accetta, R. Baron, W. Bolosky, D. Golub, (SOSP ’03), pages 29–43, Bolton Landing, NY, R. Rashid, A. Tevanian, and M. Young. Mach: October 2003. ACM SIGOPS. A new kernel foundation for UNIX development. [14] Hermann Hartig, Michael Hohmuth, Jochen In Proceedings of the Summer USENIX Techni- Liedtke, Jean Wolter, and Sebastian Schonberg. cal Conference, pages 93–112, Atlanta, GA, June The performance of Microkernel-based systems. 1986. USENIX Association. In Proceedings of the 16th Symposium on Oper- [2] The Apache Foundation. Hadoop, January 2010. ating Systems Principles (SOSP ’97), Saint Malo, http://hadoop.apache.org. France, October 1997. ACM. [3] J. Bent, G. Gibson, G. Grider, B. McClel- [15] V. Henson, A. Ven, A. Gud, and Z. Brown. land, P. Nowoczynski, J. Nunez, M. Polte, and Chunkfs: Using Divide-and-Conquer to Improve M. Wingate. Plfs: A checkpoint filesystem for File System Reliability and Repair. In Proceed- parallel applications. Technical Report LA-UR 09- ings of the Second Workshop on Hot Topics in Sys- 02117, LANL, April 2009. http://institute.lanl.gov/ tem Dependability (HotDep 2006), Seattle, WA, plfs/. November 2006. ACM SIGOPS. [4] B. Callaghan, B. Pawlowski, and P. Staubach. NFS [16] GNU Hurd. www..org/software/hurd/hurd.html. Version 3 Protocol Specification. RFC 1813, Net- [17] Shun Ishiguro, Jun Murakami, Yoshihiro Oyama, work Working Group, June 1995. and Osamu Tatebe. Optimizing local file accesses [5] Z. Cao, V. Tarasov, H. Raman, D. Hildebrand, and for FUSE-based distributed storage. In High Per- E. Zadok. On the performance variation in mod- formance Computing, Networking, Storage and ern storage stacks. In Proceedings of the 15th Analysis (SCC), 2012 SC Companion, pages 760– USENIX Conference on File and Storage Tech- 765. IEEE, 2012. nologies (FAST), Santa Clara, CA, February/March [18] N. Joukov, A. Traeger, R. Iyer, . P. Wright, and 2017. USENIX Association. to appear. E. Zadok. Profiling via Latency [6] R. Card, T. Ts’o, and S. Tweedie. Design and im- Analysis. In Proceedings of the 7th Symposium plementation of the second extended filesystem. In on Operating Systems Design and Implementation Proceedings to the First Dutch International Sym- (OSDI 2006), pages 89–102, Seattle, WA, Novem- posium on Linux, Seattle, WA, December 1994. ber 2006. ACM SIGOPS. [7] Michael Condict, Don Bolinger, Dave Mitchell, [19] Lessfs, January 2012. www.lessfs.com. and Eamonn McManus. Microkernel Modularity with Integrated Kernel Performance. In Proceed- [20] doesn’t under- ings of the First Symposium on Operating Systems stand user-space filesystems. http: Design and Implementation (OSDI 1994), Mon- //redhatstorage.redhat.com/2011/06/28/ terey, CA, November 1994. linus-torvalds-doesnt-understand-user-space-storage/. [8] Jonathan Corbet. In defense of per-bdi writeback, [21] David Mazieres. A toolkit for user-level file sys- September 2009. http://lwn.net/Articles/354851/. tems. In Proceedings of the 2001 USENIX Annual Technical Conference, 2001. [9] B. Cornell, P. A. Dinda, and F. E. Bustamante. Wayback: A User-level [22] M. K. McKusick, W. N. Joy, S. J. Leffler, and R. S. for Linux. In Proceedings of the Annual USENIX Fabry. A fast file system for UNIX. ACM Transac- Technical Conference, FREENIX Track, pages 19– tions on Computer Systems, 2(3):181–197, August 28, Boston, MA, June 2004. USENIX Association. 1984. [10] Mathieu Desnoyers. Using the Linux ker- [23] Sumit Narayan, Rohit K Mehta, and John A nel tracepoints, 2016. https://www.kernel.org/doc/ Chandy. User space storage system stack modules Documentation/trace/tracepoints.txt. with file level control. In Proceedings of the 12th [11] Ext4 Documentation. https://www.kernel.org/doc/ Annual Linux Symposium in Ottawa, pages 189– Documentation/filesystems/ext4.txt. 196, 2010. [12] Filebench, 2016. https://github.com/filebench/ [24] Nimble’s Hybrid Storage Architecture. filebench/wiki. http://info.nimblestorage.com/rs/nimblestorage/ images/nimblestorage technology overview.pdf. [13] S. Ghemawat, H. Gobioff, and S. T. Leung. The Google file system. In Proceedings of the 19th [25] NTFS-3G. www.tuxera.com.

USENIX Association 15th USENIX Conference on File and Storage Technologies 71 [26] David Pease, Arnon Amir, Lucas Villa Real, Brian [40] C. Ungureanu, B. Atkin, A. Aranya, S. Gokhale, Biskeborn, Michael Richmond, and Atsushi Abe. S. Rago, G. Calkowski, C. Dubnicki, and A. Bohra. The linear tape file system. In 2010 IEEE 26th HydraFS: a High-Throughput File System for the Symposium on Mass Storage Systems and Tech- HYDRAstor Content-Addressable Storage Sys- nologies (MSST), 2010. tem. In Proceedings of the FAST Conference, 2010. [27] Aditya Rajgarhia and Ashish Gehani. Performance [41] . Linus vs fuse. http://ceph.com/dev-notes/ and extension of user space file systems. In 25th linus-vs-fuse/. Symposium On Applied Computing . ACM, March [42] Assar Westerlund and Johan Danielsson. Arla-a 2010. free AFS client. In Proceedings of the Annual [28] Nikolaus Rath. List of fuse based file systems (git USENIX Technical Conference, FREENIX Track, page), 2011. https://github.com/libfuse/libfuse/wiki/ New Orleans, LA, June 1998. USENIX Associa- Filesystems. tion. [29] Glusterfs. http://www.gluster.org/. [43] Felix Wiemann. List of fuse based file systems [30] F. Schmuck and R. Haskin. GPFS: A shared-disk (wiki page), 2006. https://en.wikipedia.org/wiki/ file system for large computing clusters. In Pro- Filesystem in Userspace#Example uses. ceedings of the First USENIX Conference on File [44] E. Zadok and J. Nieh. FiST: A language for stack- and Storage Technologies (FAST ’02), pages 231– able file systems. In Proceedings of the Annual 244, Monterey, CA, January 2002. USENIX Asso- USENIX Technical Conference, pages 55–70, San ciation. Diego, CA, June 2000. USENIX Association. [31] Opendedup, January 2012. www.opendedup.org. [45] ZFS for Linux, January 2016. www.-fuse.net. [32] D. Steere, J. Kistler, and M. Satyanarayanan. Ef- [46] B. Zhu, K. Li, and H. Patterson. Avoiding the disk ficient user-level file cache management on the bottleneck in the data domain deduplication file sun vnode interface. In Proceedings of the Sum- system. In Proceedings of the Sixth USENIX Con- mer USENIX Technical Conference, Anaheim, CA, ference on File and Storage Technologies (FAST June 1990. IEEE. ’08), San Jose, California, USA, 2008. [33] Swaminathan Sundararaman, Laxman Visampalli, Andrea C Arpaci-Dusseau, and Remzi H Arpaci- Dusseau. Refuse to crash with Re-FUSE. In Pro- ceedings of the sixth conference on Computer sys- tems, pages 77–90. ACM, 2011. [34] A. Sweeney, D. Doucette, W. Hu, C. Anderson, M. Nishimoto, and G. Peck. in the XFS file system. In Proceedings of the Annual USENIX Technical Conference, pages 1–14, San Diego, CA, January 1996. [35] M. Szeredi. Filesystem in Userspace. http://fuse. .net, February 2005. [36] V. Tarasov, A. Gupta, K. Sourav, S. Trehan, and E. Zadok. Terra incognita: On the practicality of user-space file systems. In HotStorage ’15: Pro- ceedings of the 7th USENIX Workshop on Hot Top- ics in Storage, Santa Clara, CA, July 2015. [37] V. Tarasov, E. Zadok, and S. Shepler. Filebench: A flexible framework for file system benchmarking. ;login: The USENIX Magazine, 41(1):6–12, March 2016. [38] L. Torvalds. splice(). Kernel Trap, 2007. http: //kerneltrap.org/node/6505. [39] Linux Torwalds. Re: [patch 0/7] overlay filesys- tem: request for inclusion. https://lkml.org/lkml/ 2011/6/9/462.

72 15th USENIX Conference on File and Storage Technologies USENIX Association