fusego/connection.go

211 lines
5.1 KiB
Go
Raw Normal View History

2015-03-24 07:19:42 +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 fuse
2015-03-24 07:23:19 +03:00
import (
"fmt"
2015-03-24 07:23:19 +03:00
"log"
"path"
"runtime"
"sync"
2015-03-24 07:23:19 +03:00
"golang.org/x/net/context"
2015-03-24 07:23:19 +03:00
"github.com/jacobsa/bazilfuse"
"github.com/jacobsa/fuse/fuseops"
)
2015-03-24 07:19:42 +03:00
// A connection to the fuse kernel process.
type Connection struct {
logger *log.Logger
wrapped *bazilfuse.Conn
opsInFlight sync.WaitGroup
2015-04-29 04:53:17 +03:00
// The context from which all op contexts inherit.
parentCtx context.Context
2015-04-29 04:53:17 +03:00
// For logging purposes only.
nextOpID uint32
2015-05-05 03:41:09 +03:00
mu sync.Mutex
// A map from bazilfuse request ID (*not* the op ID for logging used above)
// to a function that cancel's its associated context.
//
// GUARDED_BY(mu)
cancelFuncs map[bazilfuse.RequestID]func()
2015-03-24 07:19:42 +03:00
}
2015-03-24 07:34:50 +03:00
// Responsibility for closing the wrapped connection is transferred to the
// result. You must call c.close() eventually.
func newConnection(
parentCtx context.Context,
2015-03-24 07:34:50 +03:00
logger *log.Logger,
2015-03-24 07:36:09 +03:00
wrapped *bazilfuse.Conn) (c *Connection, err error) {
c = &Connection{
2015-05-05 03:42:17 +03:00
logger: logger,
wrapped: wrapped,
parentCtx: parentCtx,
cancelFuncs: make(map[bazilfuse.RequestID]func()),
2015-03-24 07:36:09 +03:00
}
return
}
2015-03-24 07:23:19 +03:00
2015-04-29 04:53:17 +03:00
// Log information for an operation with the given ID. calldepth is the depth
// to use when recovering file:line information with runtime.Caller.
func (c *Connection) log(
2015-04-29 04:53:17 +03:00
opID uint32,
calldepth int,
format string,
v ...interface{}) {
// Get file:line info.
var file string
var line int
var ok bool
_, file, line, ok = runtime.Caller(calldepth)
if !ok {
file = "???"
}
2015-04-29 05:11:34 +03:00
fileLine := fmt.Sprintf("%v:%v", path.Base(file), line)
// Format the actual message to be printed.
msg := fmt.Sprintf(
2015-04-29 05:11:34 +03:00
"Op 0x%08x %24s] %v",
2015-04-29 04:51:25 +03:00
opID,
2015-04-29 05:11:34 +03:00
fileLine,
fmt.Sprintf(format, v...))
// Print it.
c.logger.Println(msg)
}
2015-05-05 03:41:09 +03:00
// LOCKS_EXCLUDED(c.mu)
func (c *Connection) recordCancelFunc(
reqID bazilfuse.RequestID,
2015-05-05 03:41:52 +03:00
f func()) {
c.mu.Lock()
defer c.mu.Unlock()
if _, ok := c.cancelFuncs[reqID]; ok {
panic(fmt.Sprintf("Already have cancel func for request %v", reqID))
}
c.cancelFuncs[reqID] = f
}
2015-05-05 03:41:09 +03:00
// Set up state for an op that is about to be returned to the user, given its
// bazilfuse request ID.
//
// Return a context that should be used for the op.
2015-05-05 03:41:09 +03:00
//
// LOCKS_EXCLUDED(c.mu)
func (c *Connection) beginOp(reqID bazilfuse.RequestID) (ctx context.Context) {
2015-05-05 03:41:09 +03:00
// Note that the op is in flight.
2015-05-05 03:04:31 +03:00
c.opsInFlight.Add(1)
2015-05-05 03:41:09 +03:00
// Set up a cancellation function.
ctx, cancel := context.WithCancel(c.parentCtx)
c.recordCancelFunc(reqID, cancel)
return
2015-05-05 03:04:31 +03:00
}
2015-05-05 03:04:03 +03:00
// Clean up all state associated with an op to which the user has responded.
2015-05-05 03:41:09 +03:00
//
// LOCKS_EXCLUDED(c.mu)
func (c *Connection) finishOp(reqID bazilfuse.RequestID) {
2015-05-05 03:41:09 +03:00
c.mu.Lock()
defer c.mu.Unlock()
// Even though the op is finished, context.WithCancel requires us to arrange
// for the cancellation function to be invoked. We also must remove it from
// our map.
cancel, ok := c.cancelFuncs[reqID]
if !ok {
panic(fmt.Sprintf("Unknown request ID in finishOp: %v", reqID))
}
cancel()
delete(c.cancelFuncs, reqID)
// Decrement the in-flight counter.
2015-05-05 03:04:31 +03:00
c.opsInFlight.Done()
}
2015-05-05 03:04:03 +03:00
2015-03-24 07:19:42 +03:00
// Read the next op from the kernel process. Return io.EOF if the kernel has
// closed the connection.
2015-03-24 07:23:19 +03:00
//
// This function delivers ops in exactly the order they are received from
2015-04-29 04:28:16 +03:00
// /dev/fuse. It must not be called multiple times concurrently.
2015-05-05 03:41:09 +03:00
//
// LOCKS_EXCLUDED(c.mu)
2015-03-24 07:26:02 +03:00
func (c *Connection) ReadOp() (op fuseops.Op, err error) {
// Keep going until we find a request we know how to convert.
for {
// Read a bazilfuse request.
2015-05-05 03:36:38 +03:00
var bfReq bazilfuse.Request
2015-03-24 07:26:02 +03:00
bfReq, err = c.wrapped.ReadRequest()
2015-05-05 03:36:38 +03:00
2015-03-24 07:26:02 +03:00
if err != nil {
return
}
// Choose an ID for this operation.
opID := c.nextOpID
c.nextOpID++
// Log the receipt of the operation.
c.log(opID, 1, "<- %v", bfReq)
2015-03-24 07:52:14 +03:00
2015-03-24 07:51:36 +03:00
// Special case: responding to this is required to make mounting work on OS
// X. We don't currently expose the capability for the file system to
// intercept this.
if statfsReq, ok := bfReq.(*bazilfuse.StatfsRequest); ok {
c.log(opID, 1, "-> (Statfs) OK")
2015-03-24 07:51:36 +03:00
statfsReq.Respond(&bazilfuse.StatfsResponse{})
continue
}
2015-05-05 03:36:38 +03:00
// Set up op dependencies.
var reqID bazilfuse.RequestID = bfReq.Hdr().ID
opCtx := c.beginOp(reqID)
logForOp := func(calldepth int, format string, v ...interface{}) {
c.log(opID, calldepth+1, format, v...)
2015-04-29 04:35:28 +03:00
}
2015-05-05 03:36:38 +03:00
finished := func(err error) { c.finishOp(reqID) }
2015-05-05 03:04:03 +03:00
2015-05-05 03:36:38 +03:00
op = fuseops.Convert(opCtx, bfReq, logForOp, finished)
2015-03-24 07:26:02 +03:00
return
}
}
2015-03-24 07:34:50 +03:00
2015-03-24 07:36:09 +03:00
func (c *Connection) waitForReady() (err error) {
<-c.wrapped.Ready
err = c.wrapped.MountError
return
}
2015-03-24 07:34:50 +03:00
// Close the connection and wait for in-flight ops.
2015-03-24 07:36:09 +03:00
func (c *Connection) close() (err error) {
err = c.wrapped.Close()
c.opsInFlight.Wait()
2015-03-24 07:36:09 +03:00
return
}