func (fs *memFS) CreateFile( op *fuseops.CreateFileOp) (err error) { fs.mu.Lock() defer fs.mu.Unlock() // Grab the parent, which we will update shortly. parent := fs.getInodeOrDie(op.Parent) // Ensure that the name doesn't already exist, so we don't wind up with a // duplicate. _, _, exists := parent.LookUpChild(op.Name) if exists { err = fuse.EEXIST return } // Set up attributes from the child, using the credentials of the calling // process as owner (matching inode_init_owner, cf. http://goo.gl/5qavg8). now := fs.clock.Now() childAttrs := fuseops.InodeAttributes{ Nlink: 1, Mode: op.Mode, Atime: now, Mtime: now, Ctime: now, Crtime: now, Uid: op.Header().Uid, Gid: op.Header().Gid, } // Allocate a child. childID, child := fs.allocateInode(childAttrs) // Add an entry in the parent. parent.AddChild(childID, op.Name, fuseutil.DT_File) // Fill in the response entry. op.Entry.Child = childID op.Entry.Attributes = child.attrs // We don't spontaneously mutate, so the kernel can cache as long as it wants // (since it also handles invalidation). op.Entry.AttributesExpiration = fs.clock.Now().Add(365 * 24 * time.Hour) op.Entry.EntryExpiration = op.Entry.EntryExpiration // We have nothing interesting to put in the Handle field. return }
// LOCKS_EXCLUDED(fs.mu) func (fs *fileSystem) CreateFile( op *fuseops.CreateFileOp) (err error) { // Find the parent. fs.mu.Lock() parent := fs.inodes[op.Parent].(inode.DirInode) fs.mu.Unlock() // Create an empty backing object for the child, failing if it already // exists. parent.Lock() o, err := parent.CreateChildFile(op.Context(), op.Name) parent.Unlock() // Special case: *gcs.PreconditionError means the name already exists. if _, ok := err.(*gcs.PreconditionError); ok { err = fuse.EEXIST return } // Propagate other errors. if err != nil { err = fmt.Errorf("CreateChildFile: %v", err) return } // Attempt to create a child inode using the object we created. If we fail to // do so, it means someone beat us to the punch with a newer generation // (unlikely, so we're probably okay with failing here). fs.mu.Lock() child := fs.lookUpOrCreateInodeIfNotStale(o.Name, o) if child == nil { err = fmt.Errorf("Newly-created record is already stale") return } defer fs.unlockAndMaybeDisposeOfInode(child, &err) // Fill out the response. op.Entry.Child = child.ID() op.Entry.Attributes, err = child.Attributes(op.Context()) if err != nil { err = fmt.Errorf("Attributes: %v", err) return } return }
func (fs *fsImpl) CreateFile( ctx context.Context, op *fuseops.CreateFileOp) (err error) { fs.mu.Lock() defer fs.mu.Unlock() // Make sure the parent exists and has not been forgotten. _ = fs.findInodeByID(op.Parent) // Mint a child inode. childID := fs.nextInodeID fs.nextInodeID++ child := &inode{ attributes: fuseops.InodeAttributes{ Nlink: 0, Mode: 0777, }, } fs.inodes[childID] = child child.IncrementLookupCount() // Return an appropriate entry. op.Entry = fuseops.ChildInodeEntry{ Child: childID, Attributes: child.attributes, } return }
func (fs *Goofys) CreateFile( ctx context.Context, op *fuseops.CreateFileOp) (err error) { fs.mu.Lock() parent := fs.getInodeOrDie(op.Parent) fs.mu.Unlock() inode, fh := parent.Create(fs, &op.Name) fs.mu.Lock() defer fs.mu.Unlock() nextInode := fs.nextInodeID fs.nextInodeID++ inode.Id = nextInode fs.inodes[inode.Id] = inode fs.inodesCache[*inode.FullName] = inode op.Entry.Child = inode.Id op.Entry.Attributes = *inode.Attributes op.Entry.AttributesExpiration = time.Now().Add(fs.flags.StatCacheTTL) op.Entry.EntryExpiration = time.Now().Add(fs.flags.TypeCacheTTL) // Allocate a handle. handleID := fs.nextHandleID fs.nextHandleID++ fs.fileHandles[handleID] = fh op.Handle = handleID inode.logFuse("<-- CreateFile") return }
// LOCKS_EXCLUDED(fs.mu) func (fs *fileSystem) CreateFile( ctx context.Context, op *fuseops.CreateFileOp) (err error) { // Create the child. child, err := fs.createFile(ctx, op.Parent, op.Name, op.Mode) if err != nil { return } defer fs.unlockAndMaybeDisposeOfInode(child, &err) // Allocate a handle. fs.mu.Lock() handleID := fs.nextHandleID fs.nextHandleID++ fs.handles[handleID] = handle.NewFileHandle( child.(*inode.FileInode), fs.bucket) op.Handle = handleID fs.mu.Unlock() // Fill out the response. e := &op.Entry e.Child = child.ID() e.Attributes, e.AttributesExpiration, err = fs.getAttributes(ctx, child) if err != nil { err = fmt.Errorf("getAttributes: %v", err) return } return }