2015-09-09 06:35:36 +03:00
|
|
|
// Copyright 2015 Google Inc. All Rights Reserved.
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
//
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
|
|
|
|
package statfs
|
|
|
|
|
|
|
|
import (
|
2018-02-07 18:12:44 +03:00
|
|
|
"context"
|
2015-09-09 06:35:36 +03:00
|
|
|
"os"
|
|
|
|
"sync"
|
|
|
|
|
|
|
|
"github.com/jacobsa/fuse"
|
|
|
|
"github.com/jacobsa/fuse/fuseops"
|
|
|
|
"github.com/jacobsa/fuse/fuseutil"
|
|
|
|
)
|
|
|
|
|
|
|
|
// A file system that allows orchestrating canned responses to statfs ops, for
|
|
|
|
// testng out OS-specific statfs behavior.
|
|
|
|
//
|
|
|
|
// The file system allows opening and writing to any name that is a child of
|
|
|
|
// the root inode, and keeps track of the most recent write size delivered by
|
|
|
|
// the kernel (in order to test statfs response block size effects on write
|
|
|
|
// size, if any).
|
|
|
|
//
|
|
|
|
// Safe for concurrent access.
|
|
|
|
type FS interface {
|
|
|
|
fuseutil.FileSystem
|
|
|
|
|
|
|
|
// Set the canned response to be used for future statfs ops.
|
|
|
|
SetStatFSResponse(r fuseops.StatFSOp)
|
|
|
|
|
2015-10-03 00:31:06 +03:00
|
|
|
// Set the canned response to be used for future stat ops.
|
|
|
|
SetStatResponse(r fuseops.InodeAttributes)
|
|
|
|
|
2015-09-09 06:35:36 +03:00
|
|
|
// Return the size of the most recent write delivered by the kernel, or -1 if
|
|
|
|
// none.
|
|
|
|
MostRecentWriteSize() int
|
|
|
|
}
|
|
|
|
|
2020-01-28 12:10:08 +03:00
|
|
|
func New() FS {
|
|
|
|
return &statFS{
|
2015-10-03 00:31:06 +03:00
|
|
|
cannedStatResponse: fuseops.InodeAttributes{
|
|
|
|
Mode: 0666,
|
|
|
|
},
|
2015-09-09 06:35:36 +03:00
|
|
|
mostRecentWriteSize: -1,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const childInodeID = fuseops.RootInodeID + 1
|
|
|
|
|
|
|
|
type statFS struct {
|
|
|
|
fuseutil.NotImplementedFileSystem
|
|
|
|
|
|
|
|
mu sync.Mutex
|
2015-10-03 00:31:06 +03:00
|
|
|
cannedResponse fuseops.StatFSOp // GUARDED_BY(mu)
|
|
|
|
cannedStatResponse fuseops.InodeAttributes // GUARDED_BY(mu)
|
|
|
|
mostRecentWriteSize int // GUARDED_BY(mu)
|
2015-09-09 06:35:36 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
|
|
// Helpers
|
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
func dirAttrs() fuseops.InodeAttributes {
|
|
|
|
return fuseops.InodeAttributes{
|
|
|
|
Mode: os.ModeDir | 0777,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-10-03 00:31:06 +03:00
|
|
|
func (fs *statFS) fileAttrs() fuseops.InodeAttributes {
|
|
|
|
return fs.cannedStatResponse
|
2015-09-09 06:35:36 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
|
|
// Public interface
|
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
// LOCKS_EXCLUDED(fs.mu)
|
|
|
|
func (fs *statFS) SetStatFSResponse(r fuseops.StatFSOp) {
|
|
|
|
fs.mu.Lock()
|
|
|
|
defer fs.mu.Unlock()
|
|
|
|
|
|
|
|
fs.cannedResponse = r
|
|
|
|
}
|
|
|
|
|
2015-10-03 00:31:06 +03:00
|
|
|
// LOCKS_EXCLUDED(fs.mu)
|
|
|
|
func (fs *statFS) SetStatResponse(r fuseops.InodeAttributes) {
|
|
|
|
fs.mu.Lock()
|
|
|
|
defer fs.mu.Unlock()
|
|
|
|
|
|
|
|
fs.cannedStatResponse = r
|
|
|
|
}
|
|
|
|
|
2015-09-09 06:35:36 +03:00
|
|
|
// LOCKS_EXCLUDED(fs.mu)
|
|
|
|
func (fs *statFS) MostRecentWriteSize() int {
|
|
|
|
fs.mu.Lock()
|
|
|
|
defer fs.mu.Unlock()
|
|
|
|
|
|
|
|
return fs.mostRecentWriteSize
|
|
|
|
}
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
|
|
// FileSystem methods
|
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2015-09-09 06:42:15 +03:00
|
|
|
// LOCKS_EXCLUDED(fs.mu)
|
|
|
|
func (fs *statFS) StatFS(
|
2015-09-09 06:35:36 +03:00
|
|
|
ctx context.Context,
|
2020-01-28 12:10:08 +03:00
|
|
|
op *fuseops.StatFSOp) error {
|
2015-09-09 06:42:15 +03:00
|
|
|
fs.mu.Lock()
|
|
|
|
defer fs.mu.Unlock()
|
2015-09-09 06:35:36 +03:00
|
|
|
|
2015-09-09 06:42:15 +03:00
|
|
|
*op = fs.cannedResponse
|
2020-01-28 12:10:08 +03:00
|
|
|
return nil
|
2015-09-09 06:35:36 +03:00
|
|
|
}
|
|
|
|
|
2015-09-09 14:45:28 +03:00
|
|
|
func (fs *statFS) LookUpInode(
|
|
|
|
ctx context.Context,
|
2020-01-28 12:10:08 +03:00
|
|
|
op *fuseops.LookUpInodeOp) error {
|
2015-09-09 14:45:28 +03:00
|
|
|
// Only the root has children.
|
|
|
|
if op.Parent != fuseops.RootInodeID {
|
2020-01-28 12:10:08 +03:00
|
|
|
return fuse.ENOENT
|
2015-09-09 14:45:28 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
op.Entry.Child = childInodeID
|
2015-10-03 00:31:06 +03:00
|
|
|
op.Entry.Attributes = fs.fileAttrs()
|
2015-09-09 14:45:28 +03:00
|
|
|
|
2020-01-28 12:10:08 +03:00
|
|
|
return nil
|
2015-09-09 14:45:28 +03:00
|
|
|
}
|
|
|
|
|
2015-09-09 06:35:36 +03:00
|
|
|
func (fs *statFS) GetInodeAttributes(
|
|
|
|
ctx context.Context,
|
2020-01-28 12:10:08 +03:00
|
|
|
op *fuseops.GetInodeAttributesOp) error {
|
2015-09-09 06:35:36 +03:00
|
|
|
switch op.Inode {
|
|
|
|
case fuseops.RootInodeID:
|
|
|
|
op.Attributes = dirAttrs()
|
|
|
|
|
|
|
|
case childInodeID:
|
2015-10-03 00:31:06 +03:00
|
|
|
op.Attributes = fs.fileAttrs()
|
2015-09-09 06:35:36 +03:00
|
|
|
|
|
|
|
default:
|
2020-01-28 12:10:08 +03:00
|
|
|
return fuse.ENOENT
|
2015-09-09 06:35:36 +03:00
|
|
|
}
|
|
|
|
|
2020-01-28 12:10:08 +03:00
|
|
|
return nil
|
2015-09-09 06:35:36 +03:00
|
|
|
}
|
|
|
|
|
2015-09-09 14:45:28 +03:00
|
|
|
func (fs *statFS) SetInodeAttributes(
|
|
|
|
ctx context.Context,
|
2020-01-28 12:10:08 +03:00
|
|
|
op *fuseops.SetInodeAttributesOp) error {
|
2015-09-09 14:45:28 +03:00
|
|
|
// Ignore calls to truncate existing files when opening.
|
2020-01-28 12:10:08 +03:00
|
|
|
return nil
|
2015-09-09 14:45:28 +03:00
|
|
|
}
|
|
|
|
|
2015-09-09 06:35:36 +03:00
|
|
|
func (fs *statFS) OpenFile(
|
|
|
|
ctx context.Context,
|
2020-01-28 12:10:08 +03:00
|
|
|
op *fuseops.OpenFileOp) error {
|
|
|
|
return nil
|
2015-09-09 06:35:36 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// LOCKS_EXCLUDED(fs.mu)
|
|
|
|
func (fs *statFS) WriteFile(
|
|
|
|
ctx context.Context,
|
2020-01-28 12:10:08 +03:00
|
|
|
op *fuseops.WriteFileOp) error {
|
2015-09-09 06:35:36 +03:00
|
|
|
fs.mu.Lock()
|
|
|
|
defer fs.mu.Unlock()
|
|
|
|
|
|
|
|
fs.mostRecentWriteSize = len(op.Data)
|
2020-01-28 12:10:08 +03:00
|
|
|
return nil
|
2015-09-09 06:35:36 +03:00
|
|
|
}
|