// Copyright 2016 the Go-FUSE Authors. All rights reserved. // Use of this source code is governed by a BSD-style // license that can be found in the LICENSE file. // Package fuse provides APIs to implement filesystems in // userspace in terms of raw FUSE protocol. // // A filesystem is implemented by implementing its server that provides a // RawFileSystem interface. Typically the server embeds // NewDefaultRawFileSystem() and implements only subset of filesystem methods: // // type MyFS struct { // fuse.RawFileSystem // ... // } // // func NewMyFS() *MyFS { // return &MyFS{ // RawFileSystem: fuse.NewDefaultRawFileSystem(), // ... // } // } // // // Mkdir implements "mkdir" request handler. // // // // For other requests - not explicitly implemented by MyFS - ENOSYS // // will be typically returned to client. // func (fs *MyFS) Mkdir(...) { // ... // } // // Then the filesystem can be mounted and served to a client (typically OS // kernel) by creating Server: // // fs := NewMyFS() // implements RawFileSystem // fssrv, err := fuse.NewServer(fs, mountpoint, &fuse.MountOptions{...}) // if err != nil { // ... // } // // and letting the server do its work: // // // either synchronously - .Serve() blocks until the filesystem is unmounted. // fssrv.Serve() // // // or in the background - .Serve() is spawned in another goroutine, but // // before interacting with fssrv from current context we have to wait // // until the filesystem mounting is complete. // go fssrv.Serve() // err = fssrv.WaitMount() // if err != nil { // ... // } // // The server will serve clients by dispatching their requests to the // filesystem implementation and conveying responses back. For example "mkdir" // FUSE request dispatches to call // // fs.Mkdir(*MkdirIn, ..., *EntryOut) // // "stat" to call // // fs.GetAttr(*GetAttrIn, *AttrOut) // // etc. Please refer to RawFileSystem documentation for details. // // Typically, each call of the API happens in its own // goroutine, so take care to make the file system thread-safe. // // Be careful when you access the FUSE mount from the same process. An access can // tie up two OS threads (one on the request side and one on the FUSE server side). // This can deadlock if there is no free thread to handle the FUSE server side. // Run your program with GOMAXPROCS=1 to make the problem easier to reproduce, // see https://github.com/hanwen/go-fuse/issues/261 for an example of that // problem. // // Higher level interfaces // // As said above this packages provides way to implement filesystems in terms of // raw FUSE protocol. Additionally packages nodefs and pathfs provide ways to // implement filesystem at higher levels: // // Package github.com/hanwen/go-fuse/fuse/nodefs provides way to implement // filesystems in terms of inodes. This resembles kernel's idea of what a // filesystem looks like. // // Package github.com/hanwen/go-fuse/fuse/pathfs provides way to implement // filesystems in terms of path names. Working with path names is somewhat // easier compared to inodes, however renames can be racy. Do not use pathfs if // you care about correctness. package fuse // Types for users to implement. // The result of Read is an array of bytes, but for performance // reasons, we can also return data as a file-descriptor/offset/size // tuple. If the backing store for a file is another filesystem, this // reduces the amount of copying between the kernel and the FUSE // server. The ReadResult interface captures both cases. type ReadResult interface { // Returns the raw bytes for the read, possibly using the // passed buffer. The buffer should be larger than the return // value from Size. Bytes(buf []byte) ([]byte, Status) // Size returns how many bytes this return value takes at most. Size() int // Done() is called after sending the data to the kernel. Done() } type MountOptions struct { AllowOther bool // Options are passed as -o string to fusermount. Options []string // Default is _DEFAULT_BACKGROUND_TASKS, 12. This numbers // controls the allowed number of requests that relate to // async I/O. Concurrency for synchronous I/O is not limited. MaxBackground int // Write size to use. If 0, use default. This number is // capped at the kernel maximum. MaxWrite int // Max read ahead to use. If 0, use default. This number is // capped at the kernel maximum. MaxReadAhead int // If IgnoreSecurityLabels is set, all security related xattr // requests will return NO_DATA without passing through the // user defined filesystem. You should only set this if you // file system implements extended attributes, and you are not // interested in security labels. IgnoreSecurityLabels bool // ignoring labels should be provided as a fusermount mount option. // If RememberInodes is set, we will never forget inodes. // This may be useful for NFS. RememberInodes bool // Values shown in "df -T" and friends // First column, "Filesystem" FsName string // Second column, "Type", will be shown as "fuse." + Name Name string // If set, wrap the file system in a single-threaded locking wrapper. SingleThreaded bool // If set, return ENOSYS for Getxattr calls, so the kernel does not issue any // Xattr operations at all. DisableXAttrs bool // If set, print debugging information. Debug bool // If set, ask kernel to forward file locks to FUSE. If using, // you must implement the GetLk/SetLk/SetLkw methods. EnableLocks bool // If set, ask kernel not to do automatic data cache invalidation. // The filesystem is fully responsible for invalidating data cache. ExplicitDataCacheControl bool // If set, fuse will first attempt to use syscall.Mount instead of // fusermount to mount the filesystem. This will not update /etc/mtab // but might be needed if fusermount is not available. DirectMount bool // Options passed to syscall.Mount, the default value used by fusermount // is syscall.MS_NOSUID|syscall.MS_NODEV DirectMountFlags uintptr } // RawFileSystem is an interface close to the FUSE wire protocol. // // Unless you really know what you are doing, you should not implement // this, but rather the nodefs.Node or pathfs.FileSystem interfaces; the // details of getting interactions with open files, renames, and threading // right etc. are somewhat tricky and not very interesting. // // Each FUSE request results in a corresponding method called by Server. // Several calls may be made simultaneously, because the server typically calls // each method in separate goroutine. // // A null implementation is provided by NewDefaultRawFileSystem. // // After a successful FUSE API call returns, you may not read input or // write output data: for performance reasons, memory is reused for // following requests, and reading/writing the request data will lead // to race conditions. If you spawn a background routine from a FUSE // API call, any incoming request data it wants to reference should be // copied over. // // If a FUSE API call is canceled (which is signaled by closing the // `cancel` channel), the API call should return EINTR. In this case, // the outstanding request data is not reused, so the API call may // return EINTR without ensuring that child contexts have successfully // completed. type RawFileSystem interface { String() string // If called, provide debug output through the log package. SetDebug(debug bool) // Lookup is called by the kernel when the VFS wants to know // about a file inside a directory. Many lookup calls can // occur in parallel, but only one call happens for each (dir, // name) pair. Lookup(cancel <-chan struct{}, header *InHeader, name string, out *EntryOut) (status Status) // Forget is called when the kernel discards entries from its // dentry cache. This happens on unmount, and when the kernel // is short on memory. Since it is not guaranteed to occur at // any moment, and since there is no return value, Forget // should not do I/O, as there is no channel to report back // I/O errors. Forget(nodeid, nlookup uint64) // Attributes. GetAttr(cancel <-chan struct{}, input *GetAttrIn, out *AttrOut) (code Status) SetAttr(cancel <-chan struct{}, input *SetAttrIn, out *AttrOut) (code Status) // Modifying structure. Mknod(cancel <-chan struct{}, input *MknodIn, name string, out *EntryOut) (code Status) Mkdir(cancel <-chan struct{}, input *MkdirIn, name string, out *EntryOut) (code Status) Unlink(cancel <-chan struct{}, header *InHeader, name string) (code Status) Rmdir(cancel <-chan struct{}, header *InHeader, name string) (code Status) Rename(cancel <-chan struct{}, input *RenameIn, oldName string, newName string) (code Status) Link(cancel <-chan struct{}, input *LinkIn, filename string, out *EntryOut) (code Status) Symlink(cancel <-chan struct{}, header *InHeader, pointedTo string, linkName string, out *EntryOut) (code Status) Readlink(cancel <-chan struct{}, header *InHeader) (out []byte, code Status) Access(cancel <-chan struct{}, input *AccessIn) (code Status) // Extended attributes. // GetXAttr reads an extended attribute, and should return the // number of bytes. If the buffer is too small, return ERANGE, // with the required buffer size. GetXAttr(cancel <-chan struct{}, header *InHeader, attr string, dest []byte) (sz uint32, code Status) // ListXAttr lists extended attributes as '\0' delimited byte // slice, and return the number of bytes. If the buffer is too // small, return ERANGE, with the required buffer size. ListXAttr(cancel <-chan struct{}, header *InHeader, dest []byte) (uint32, Status) // SetAttr writes an extended attribute. SetXAttr(cancel <-chan struct{}, input *SetXAttrIn, attr string, data []byte) Status // RemoveXAttr removes an extended attribute. RemoveXAttr(cancel <-chan struct{}, header *InHeader, attr string) (code Status) // File handling. Create(cancel <-chan struct{}, input *CreateIn, name string, out *CreateOut) (code Status) Open(cancel <-chan struct{}, input *OpenIn, out *OpenOut) (status Status) Read(cancel <-chan struct{}, input *ReadIn, buf []byte) (ReadResult, Status) Lseek(cancel <-chan struct{}, in *LseekIn, out *LseekOut) Status // File locking GetLk(cancel <-chan struct{}, input *LkIn, out *LkOut) (code Status) SetLk(cancel <-chan struct{}, input *LkIn) (code Status) SetLkw(cancel <-chan struct{}, input *LkIn) (code Status) Release(cancel <-chan struct{}, input *ReleaseIn) Write(cancel <-chan struct{}, input *WriteIn, data []byte) (written uint32, code Status) CopyFileRange(cancel <-chan struct{}, input *CopyFileRangeIn) (written uint32, code Status) Flush(cancel <-chan struct{}, input *FlushIn) Status Fsync(cancel <-chan struct{}, input *FsyncIn) (code Status) Fallocate(cancel <-chan struct{}, input *FallocateIn) (code Status) // Directory handling OpenDir(cancel <-chan struct{}, input *OpenIn, out *OpenOut) (status Status) ReadDir(cancel <-chan struct{}, input *ReadIn, out *DirEntryList) Status ReadDirPlus(cancel <-chan struct{}, input *ReadIn, out *DirEntryList) Status ReleaseDir(input *ReleaseIn) FsyncDir(cancel <-chan struct{}, input *FsyncIn) (code Status) StatFs(cancel <-chan struct{}, input *InHeader, out *StatfsOut) (code Status) // This is called on processing the first request. The // filesystem implementation can use the server argument to // talk back to the kernel (through notify methods). Init(*Server) }