forked from hanwen/go-fuse
-
Notifications
You must be signed in to change notification settings - Fork 1
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
darwin: use simpleHandleMap that never reuses handle numbers
Implement simpleHandleMap that never reuses handle numbers and uses a simple map to track handle numbers. This seems to get rid of the osxfuse: vnode changed generation errors we are seeing on darwin, while sacrificing some performance. I have tested simpleHandleMap on darwin and linux, seems to work fine. rfjakob/gocryptfs#213 hanwen#204 macfuse/macfuse#482
- Loading branch information
Showing
6 changed files
with
141 additions
and
16 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,15 @@ | ||
// Copyright 2018 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 nodefs | ||
|
||
func init() { | ||
// darwin / osxfuse has problems with handle reuse, we get | ||
// the kernel message | ||
// osxfuse: vnode changed generation | ||
// and errors returned to the user. simpleHandleMap never reuses handles | ||
// which seems to keep osxfuse happy. | ||
// https://github.com/hanwen/go-fuse/issues/204 | ||
newHandleMap = newSimpleHandleMap | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,97 @@ | ||
// Copyright 2018 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 nodefs | ||
|
||
import ( | ||
"log" | ||
"sync" | ||
) | ||
|
||
// simpleHandleMap is a simple, low-performance handle map that never reuses | ||
// handle numbers. | ||
type simpleHandleMap struct { | ||
sync.Mutex | ||
entries map[uint64]*handled | ||
nextHandle uint64 | ||
} | ||
|
||
func newSimpleHandleMap() handleMap { | ||
return &simpleHandleMap{ | ||
entries: make(map[uint64]*handled), | ||
// Avoid handing out ID 0 and 1, like portableHandleMap does. | ||
nextHandle: 2, | ||
} | ||
} | ||
|
||
func (s *simpleHandleMap) Register(obj *handled) (handle, generation uint64) { | ||
s.Lock() | ||
defer s.Unlock() | ||
// The object already has a handle | ||
if obj.count != 0 { | ||
if obj.handle == 0 { | ||
log.Panicf("bug: count=%d handle=%d", obj.count, obj.handle) | ||
} | ||
obj.count++ | ||
return obj.handle, 0 | ||
} | ||
// Create a new handle | ||
obj.count = 1 | ||
obj.handle = s.nextHandle | ||
s.entries[s.nextHandle] = obj | ||
s.nextHandle++ | ||
return obj.handle, 0 | ||
} | ||
|
||
// Count returns the number of currently used handles | ||
func (s *simpleHandleMap) Count() int { | ||
s.Lock() | ||
defer s.Unlock() | ||
return len(s.entries) | ||
} | ||
|
||
// Handle gets the object's uint64 handle. | ||
func (s *simpleHandleMap) Handle(obj *handled) (handle uint64) { | ||
s.Lock() | ||
defer s.Unlock() | ||
if obj.count == 0 { | ||
return 0 | ||
} | ||
return obj.handle | ||
} | ||
|
||
// Decode retrieves a stored object from its uint64 handle. | ||
func (s *simpleHandleMap) Decode(handle uint64) *handled { | ||
s.Lock() | ||
defer s.Unlock() | ||
return s.entries[handle] | ||
} | ||
|
||
// Forget decrements the reference counter for "handle" by "count" and drops | ||
// the object if the refcount reaches zero. | ||
// Returns a boolean whether the object was dropped and the object itself. | ||
func (s *simpleHandleMap) Forget(handle uint64, count int) (forgotten bool, obj *handled) { | ||
s.Lock() | ||
defer s.Unlock() | ||
obj = s.entries[handle] | ||
obj.count -= count | ||
if obj.count < 0 { | ||
log.Panicf("underflow: handle %d, count %d, obj.count %d", handle, count, obj.count) | ||
} | ||
if obj.count > 0 { | ||
return false, obj | ||
} | ||
// count is zero, drop the reference | ||
delete(s.entries, handle) | ||
obj.handle = 0 | ||
return true, obj | ||
} | ||
|
||
// Has checks if the uint64 handle is stored. | ||
func (s *simpleHandleMap) Has(handle uint64) bool { | ||
s.Lock() | ||
defer s.Unlock() | ||
_, ok := s.entries[handle] | ||
return ok | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters