12 KiB
Technical Behavior and Limitations
Do hard links work?
Yes. links (otherwise known as hard links) are fundamentally supported by mergerfs. There is no option to enable or disable links. All comments elswhere claiming they do not work are related to the setup of the user's system.
See the Intro to Filesystems page for more
details on what a link
is.
A limitation of a link
is that it must reference a file/inode
within the same filesystem or device. When a request is made to the
operating system to link
(or rename
) a file to a target location
the OS will first check if the source and target are the same
filesystem / device. If not it will return the error EXDEV
meaning
that the request is not able to be made due to the result being "cross
device".
What catches many users off guard is that this "cross device" check is not very deep or involved. Even the same filesystem bind mounted to another location will typically be considered a different device. So when mergerfs is creating a union of other filesystems there is no way to link or rename from one of the underlying branches into mergerfs. They are different filesystems even if logically they seem related or the same. And similarly it is impossible for mergerfs to truly make a link between two branches which are not in fact the same mounted filesystem.
This is relevant to mergerfs users because bind mounts are very common
when using container runtimes such as Docker, Podman, or
Kubernetes. When you add a volume or mount to your container that is
in fact a bind mount of that path. Meaning if you were to add
/mnt/mergerfs/downloads
and /mnt/mergerfs/movies
as separate
mounts into your container any request to link between the two
locations within the container will fail with the EXDEV
error even
though it will work perfectly fine on the host. If you wish for link
and rename
to work you MUST mount into the container a point in
the directory structure above what you wish to use. In this case
/mnt/mergerfs
.
Another situation where people will hit unexpected EXDEV
errors is
when they use a policy which makes file creation decisions based on
existing paths. If the source of a link
or rename
is on branch A
but the target directory is on branch B, given the whole point of the
policy is to strictly enforce behavior based on what branches already
exist, it must return EXDEV
. Doing otherwise would fundamentally
break the purpose of the policy. Read rename &
link for more details on how mergerfs
handles those two functions.
When in doubt it is trivial to test link
:
$ cd /mnt/mergerfs/
$ touch foo
$ ln -v foo bar
'bar' => 'foo'
$ stat foo bar | grep Inode
Device: 0,173 Inode: 123456 Links: 2
Device: 0,173 Inode: 123456 Links: 2
For the more technical: within mergerfs the inode of a file is not necessarily indicative of two file names linking to the same underlying data. For more details as to why that is see the docs on inodecalc.
Do reflink, FICLONE, or FICLONERANGE work?
Unfortunately not. FUSE, the technology mergerfs is based on, does not
support the clone_file_range
feature needed for it to work. mergerfs
won't even know such a request is made. The kernel will simply return
an error back to the application making the request.
Should FUSE gain the ability mergerfs will be updated to support it.
That said: mergerfs itself will attempt to use FICLONE
when copying
data between files in the limited cases where it copies/moves files.
How does mergerfs handle moving and copying of files?
This is a very common mistaken assumption regarding how filesystems work. There is no such thing as "move" or "copy." These concepts are high level behaviors made up of numerous independent steps and not individual filesystem functions.
A "move" can include a "copy" so lets describe copy first.
When an application copies a file from "source" to "destination" it can do so in a number of ways but the basics are the following.
open
the source file.create
the destination file.read
a chunk of data from source andwrite
to destination. Continue till it runs out of data to copy.- Copy file metadata (
stat
) such as ownership (chown
), permissions (chmod
), timestamps (utimes
), extended attributes (getxattr
,setxattr
), etc. close
source and destination files.
"move" is typically a rename(src,dst)
and if that errors with
EXDEV
(meaning the source and destination are on different
filesystems) the application will "copy" the file as described above
and then it removes (unlink
) the source.
The rename(src,dst)
, open(src)
, create(dst)
, data copying,
metadata copying, unlink(src)
, etc. are entirely distinct and
separate events. There is really no practical way to know that what is
ultimately occurring is the "copying" of a file or what the source
file would be. Since the source is not known there is no way to know
how large a created file is destined to become. This is why it is
impossible for mergerfs to choose the branch for a create
based on
file size. The only context provided when a file is created, besides
the name, is the permissions, if it is to be read and/or written, and
some low level settings for the operating system.
All of this means that mergerfs can not make decisions when a file is created based on file size or the source of the data. That information is simply not available. At best mergerfs could respond to files reaching a certain size when writing data, a file is closed, or renamed.
Related: if a user wished to have mergerfs perform certain activities based on the name of a file it is common and even best practice for a program to write to a temporary file first and then rename to its final destination. That temporary file name will typically be random and have no indication of the type of file being written. At best something could be done on rename.
Additional reading: Intro to Filesystems: Workflows
Why does the total available space in mergerfs not equal outside?
Are you using ext2/3/4? With reserve for root? mergerfs uses available space for statfs calculations. If you've reserved space for root then it won't show up.
You can remove the reserve by running: tune2fs -m 0 <device>
I notice massive slowdowns of writes when enabling cache.files.
When file caching is enabled in any form (cache.files!=off
) it will
issue getxattr
requests for security.capability
prior to every
single write. This will usually result in performance degradation,
especially when using a network filesystem (such as NFS or SMB.)
Unfortunately at this moment, the kernel is not caching the response.
To work around this situation mergerfs offers a few solutions.
- Set
security_capability=false
. It will short circuit any call and returnENOATTR
. This still means though that mergerfs will receive the request before every write but at least it doesn't get passed through to the underlying filesystem. - Set
xattr=noattr
. Same as above but applies to all calls to getxattr. Not justsecurity.capability
. This will not be cached by the kernel either but mergerfs' runtime config system will still function. - Set
xattr=nosys
. Results in mergerfs returningENOSYS
which will be cached by the kernel. No future xattr calls will be forwarded to mergerfs. The downside is that also means the xattr based config and query functionality won't work either. - Disable file caching. If you aren't using applications which use
mmap
it's probably simpler to just disable it altogether. The kernel won't send the requests when caching is disabled.
Why use FUSE? Why not a kernel based solution?
As with any solution to a problem, there are advantages and disadvantages to each one.
A FUSE based solution has all the downsides of FUSE:
- Higher IO latency due to the trips in and out of kernel space
- Higher general overhead due to trips in and out of kernel space
- Double caching when using page caching
- Misc limitations due to FUSE's design
But FUSE also has a lot of upsides:
- Easier to offer a cross platform solution
- Easier forward and backward compatibility
- Easier updates for users
- Easier and faster release cadence
- Allows more flexibility in design and features
- Overall easier to write, secure, and maintain
- Much lower barrier to entry (getting code into the kernel takes a lot of time and effort initially)
Is my OS's libfuse needed for mergerfs to work?
No. Normally mount.fuse
is needed to get mergerfs (or any FUSE
filesystem to mount using the mount
command but in vendoring the
libfuse library the mount.fuse
app has been renamed to
mount.mergerfs
meaning the filesystem type in fstab
can simply be
mergerfs
. That said there should be no harm in having it installed
and continuing to using fuse.mergerfs
as the type in /etc/fstab
.
If mergerfs
doesn't work as a type it could be due to how the
mount.mergerfs
tool was installed. Must be in /sbin/
with proper
permissions.
Why was splice support removed?
After a lot of testing over the years, splicing always appeared to at best, provide equivalent performance, and in some cases, worse performance. Splice is not supported on other platforms forcing a traditional read/write fallback to be provided. The splice code was removed to simplify the codebase.
How does mergerfs handle credentials?
mergerfs is a multithreaded application in order to handle requests from the kernel concurrently. Each FUSE message has a header with certain details about the request including the process ID (pid) of the requesting application, the process' effective user id (uid), and group id (gid). To ensure proper POSIX filesystem behavior and security mergerfs must change its identity to match that of the requester when performing the certain functions on the underlying filesystem. As required by standards most Unix/POSIX based systems a process and all its threads are under the same uid and gid. However, on Linux each thread may have its own credentials. This allows mergerfs to be multithreaded and for each thread to change to the credentials as required by the incoming message it is handling. However, currently on FreeBSD this is not possible (though there has been discussions) and as such must change the credentials of the whole application when actioning messages. mergerfs does optimize this behavior by only changing credentials and locking the thread to do so if the process is currently not the same as what is necessary by the incoming request. As a result of this design FreeBSD may experience more contention and therefore lower performance than Linux.
Additionally, mergerfs utilizes a cache for supplemental groups due the the high cost of querying that information.
Does mergerfs support idmap?
Yes. At least in so far as it's been enabled now the FUSE itself allows a filesystem to indicate it is allowed.
Requires that kernel-permissions-check be enabled (the default.)
If there are any usage issues contact the author.
What happens if a branch filesystem blocks?
POSIX filesystem calls unfortunately are entirely synchronous. If the underlying filesystem mergerfs is interacting with freezes up or blocks then the thread issuing the request will block same as any other piece of software. If enough threads block then mergerfs will block. There are no timeouts or ways to truly work around this situation. If mergerfs spawned new threads it is very likely they too would end up blocked.
What information does mergerfs have when making decisions?
Every incoming request contains:
- The
tid
orpid
of the calling thread/process - The
uid
andgid
of the calling thread/process - The
umask
of the calling thread/process - The arguments of the filesystem call in question
Naturally the mergerfs config as well as anything queriable from the operating system or filesystems are also available.