Files

131 lines
3.8 KiB
Go
Raw Permalink Normal View History

2023-11-10 10:20:15 -08:00
// Copyright 2023 The gVisor Authors.
//
// 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 devutil provides device specific utilities.
package devutil
import (
2023-11-10 14:13:03 -08:00
"fmt"
2023-11-10 10:20:15 -08:00
"golang.org/x/sys/unix"
"gvisor.dev/gvisor/pkg/context"
2023-11-10 14:13:03 -08:00
"gvisor.dev/gvisor/pkg/fsutil"
2023-11-10 10:20:15 -08:00
"gvisor.dev/gvisor/pkg/lisafs"
2023-11-10 14:13:03 -08:00
"gvisor.dev/gvisor/pkg/log"
2023-11-10 10:20:15 -08:00
"gvisor.dev/gvisor/pkg/unet"
)
// GoferClient is the lisafs client for the /dev gofer connection.
type GoferClient struct {
clientFD lisafs.ClientFD
hostFD int
2024-05-04 03:07:25 -07:00
contName string
2023-11-10 10:20:15 -08:00
}
// NewGoferClient establishes the LISAFS connection to the dev gofer server.
2024-05-04 03:07:25 -07:00
// It takes ownership of fd. contName is the owning container name.
func NewGoferClient(ctx context.Context, contName string, fd int) (*GoferClient, error) {
2023-11-10 10:20:15 -08:00
ctx.UninterruptibleSleepStart(false)
defer ctx.UninterruptibleSleepFinish(false)
sock, err := unet.NewSocket(fd)
if err != nil {
ctx.Warningf("failed to create socket for dev gofer client: %v", err)
return nil, err
}
client, devInode, devHostFD, err := lisafs.NewClient(sock)
if err != nil {
ctx.Warningf("failed to create dev gofer client: %v", err)
return nil, err
}
return &GoferClient{
clientFD: client.NewFD(devInode.ControlFD),
hostFD: devHostFD,
2024-05-08 00:06:21 -07:00
contName: contName,
2023-11-10 10:20:15 -08:00
}, nil
}
// Close closes the LISAFS connection.
func (g *GoferClient) Close() {
// Close the connection to the server. This implicitly closes all FDs.
g.clientFD.Client().Close()
if g.hostFD >= 0 {
_ = unix.Close(g.hostFD)
}
}
2023-11-10 14:13:03 -08:00
2024-05-04 03:07:25 -07:00
// ContainerName returns the name of the container that owns this gofer.
func (g *GoferClient) ContainerName() string {
return g.contName
}
2023-11-10 14:13:03 -08:00
// DirentNames returns names of all the dirents for /dev on the gofer.
func (g *GoferClient) DirentNames(ctx context.Context) ([]string, error) {
if g.hostFD >= 0 {
return fsutil.DirentNames(g.hostFD)
}
client := g.clientFD.Client()
openFDID, _, err := g.clientFD.OpenAt(ctx, unix.O_RDONLY)
if err != nil {
return nil, fmt.Errorf("failed to open dev from gofer: %v", err)
}
defer client.CloseFD(ctx, openFDID, true /* flush */)
openFD := client.NewFD(openFDID)
const count = int32(64 * 1024)
var names []string
for {
dirents, err := openFD.Getdents64(ctx, count)
if err != nil {
return nil, fmt.Errorf("Getdents64 RPC failed: %v", err)
}
if len(dirents) == 0 {
break
}
for i := range dirents {
names = append(names, string(dirents[i].Name))
}
}
return names, nil
}
// OpenAt opens the device file at /dev/{name} on the gofer.
func (g *GoferClient) OpenAt(ctx context.Context, name string, flags uint32) (int, error) {
flags &= unix.O_ACCMODE
if g.hostFD >= 0 {
return unix.Openat(g.hostFD, name, int(flags|unix.O_NOFOLLOW), 0)
}
childInode, err := g.clientFD.Walk(ctx, name)
if err != nil {
log.Infof("failed to walk %q from dev gofer FD", name)
return 0, err
}
client := g.clientFD.Client()
childFD := client.NewFD(childInode.ControlFD)
childOpenFD, childHostFD, err := childFD.OpenAt(ctx, flags)
if err != nil {
log.Infof("failed to open %q from child FD", name)
client.CloseFD(ctx, childFD.ID(), true /* flush */)
return 0, err
}
client.CloseFD(ctx, childFD.ID(), false /* flush */)
client.CloseFD(ctx, childOpenFD, true /* flush */)
return childHostFD, nil
}
2024-05-06 17:33:42 -07:00
// GoferClientProvider provides a GoferClient for a given container.
type GoferClientProvider interface {
GetDevGoferClient(contName string) *GoferClient
}