gvisor/pkg/sentry/socket/rpcinet/stack_unsafe.go

194 lines
6.0 KiB
Go

// Copyright 2018 Google LLC
//
// 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 rpcinet
import (
"syscall"
"unsafe"
"gvisor.googlesource.com/gvisor/pkg/abi/linux"
"gvisor.googlesource.com/gvisor/pkg/binary"
pb "gvisor.googlesource.com/gvisor/pkg/sentry/socket/rpcinet/syscall_rpc_go_proto"
"gvisor.googlesource.com/gvisor/pkg/sentry/usermem"
"gvisor.googlesource.com/gvisor/pkg/syserr"
)
// NewNetlinkRouteRequest builds a netlink message for getting the RIB,
// the routing information base.
func newNetlinkRouteRequest(proto, seq, family int) []byte {
rr := &syscall.NetlinkRouteRequest{}
rr.Header.Len = uint32(syscall.NLMSG_HDRLEN + syscall.SizeofRtGenmsg)
rr.Header.Type = uint16(proto)
rr.Header.Flags = syscall.NLM_F_DUMP | syscall.NLM_F_REQUEST
rr.Header.Seq = uint32(seq)
rr.Data.Family = uint8(family)
return netlinkRRtoWireFormat(rr)
}
func netlinkRRtoWireFormat(rr *syscall.NetlinkRouteRequest) []byte {
b := make([]byte, rr.Header.Len)
*(*uint32)(unsafe.Pointer(&b[0:4][0])) = rr.Header.Len
*(*uint16)(unsafe.Pointer(&b[4:6][0])) = rr.Header.Type
*(*uint16)(unsafe.Pointer(&b[6:8][0])) = rr.Header.Flags
*(*uint32)(unsafe.Pointer(&b[8:12][0])) = rr.Header.Seq
*(*uint32)(unsafe.Pointer(&b[12:16][0])) = rr.Header.Pid
b[16] = byte(rr.Data.Family)
return b
}
func (s *Stack) getNetlinkFd() (uint32, *syserr.Error) {
id, c := s.rpcConn.NewRequest(pb.SyscallRequest{Args: &pb.SyscallRequest_Socket{&pb.SocketRequest{Family: int64(syscall.AF_NETLINK), Type: int64(syscall.SOCK_RAW | syscall.SOCK_NONBLOCK), Protocol: int64(syscall.NETLINK_ROUTE)}}}, false /* ignoreResult */)
<-c
res := s.rpcConn.Request(id).Result.(*pb.SyscallResponse_Socket).Socket.Result
if e, ok := res.(*pb.SocketResponse_ErrorNumber); ok {
return 0, syserr.FromHost(syscall.Errno(e.ErrorNumber))
}
return res.(*pb.SocketResponse_Fd).Fd, nil
}
func (s *Stack) bindNetlinkFd(fd uint32, sockaddr []byte) *syserr.Error {
id, c := s.rpcConn.NewRequest(pb.SyscallRequest{Args: &pb.SyscallRequest_Bind{&pb.BindRequest{Fd: fd, Address: sockaddr}}}, false /* ignoreResult */)
<-c
if e := s.rpcConn.Request(id).Result.(*pb.SyscallResponse_Bind).Bind.ErrorNumber; e != 0 {
return syserr.FromHost(syscall.Errno(e))
}
return nil
}
func (s *Stack) closeNetlinkFd(fd uint32) {
_, _ = s.rpcConn.NewRequest(pb.SyscallRequest{Args: &pb.SyscallRequest_Close{&pb.CloseRequest{Fd: fd}}}, true /* ignoreResult */)
}
func (s *Stack) rpcSendMsg(req *pb.SyscallRequest_Sendmsg) (uint32, *syserr.Error) {
id, c := s.rpcConn.NewRequest(pb.SyscallRequest{Args: req}, false /* ignoreResult */)
<-c
res := s.rpcConn.Request(id).Result.(*pb.SyscallResponse_Sendmsg).Sendmsg.Result
if e, ok := res.(*pb.SendmsgResponse_ErrorNumber); ok {
return 0, syserr.FromHost(syscall.Errno(e.ErrorNumber))
}
return res.(*pb.SendmsgResponse_Length).Length, nil
}
func (s *Stack) sendMsg(fd uint32, buf []byte, to []byte, flags int) (int, *syserr.Error) {
// Whitelist flags.
if flags&^(syscall.MSG_DONTWAIT|syscall.MSG_EOR|syscall.MSG_FASTOPEN|syscall.MSG_MORE|syscall.MSG_NOSIGNAL) != 0 {
return 0, syserr.ErrInvalidArgument
}
req := &pb.SyscallRequest_Sendmsg{&pb.SendmsgRequest{
Fd: fd,
Data: buf,
Address: to,
More: flags&linux.MSG_MORE != 0,
EndOfRecord: flags&linux.MSG_EOR != 0,
}}
n, err := s.rpcSendMsg(req)
return int(n), err
}
func (s *Stack) rpcRecvMsg(req *pb.SyscallRequest_Recvmsg) (*pb.RecvmsgResponse_ResultPayload, *syserr.Error) {
id, c := s.rpcConn.NewRequest(pb.SyscallRequest{Args: req}, false /* ignoreResult */)
<-c
res := s.rpcConn.Request(id).Result.(*pb.SyscallResponse_Recvmsg).Recvmsg.Result
if e, ok := res.(*pb.RecvmsgResponse_ErrorNumber); ok {
return nil, syserr.FromHost(syscall.Errno(e.ErrorNumber))
}
return res.(*pb.RecvmsgResponse_Payload).Payload, nil
}
func (s *Stack) recvMsg(fd, l, flags uint32) ([]byte, *syserr.Error) {
req := &pb.SyscallRequest_Recvmsg{&pb.RecvmsgRequest{
Fd: fd,
Length: l,
Sender: false,
Trunc: flags&linux.MSG_TRUNC != 0,
Peek: flags&linux.MSG_PEEK != 0,
}}
res, err := s.rpcRecvMsg(req)
if err != nil {
return nil, err
}
return res.Data, nil
}
func (s *Stack) netlinkRequest(proto, family int) ([]byte, error) {
fd, err := s.getNetlinkFd()
if err != nil {
return nil, err.ToError()
}
defer s.closeNetlinkFd(fd)
lsa := syscall.SockaddrNetlink{Family: syscall.AF_NETLINK}
b := binary.Marshal(nil, usermem.ByteOrder, &lsa)
if err := s.bindNetlinkFd(fd, b); err != nil {
return nil, err.ToError()
}
wb := newNetlinkRouteRequest(proto, 1, family)
_, err = s.sendMsg(fd, wb, b, 0)
if err != nil {
return nil, err.ToError()
}
var tab []byte
done:
for {
rb, err := s.recvMsg(fd, uint32(syscall.Getpagesize()), 0)
nr := len(rb)
if err != nil {
return nil, err.ToError()
}
if nr < syscall.NLMSG_HDRLEN {
return nil, syserr.ErrInvalidArgument.ToError()
}
tab = append(tab, rb...)
msgs, e := syscall.ParseNetlinkMessage(rb)
if e != nil {
return nil, e
}
for _, m := range msgs {
if m.Header.Type == syscall.NLMSG_DONE {
break done
}
if m.Header.Type == syscall.NLMSG_ERROR {
return nil, syserr.ErrInvalidArgument.ToError()
}
}
}
return tab, nil
}
// DoNetlinkRouteRequest returns routing information base, also known as RIB,
// which consists of network facility information, states and parameters.
func (s *Stack) DoNetlinkRouteRequest(req int) ([]syscall.NetlinkMessage, error) {
data, err := s.netlinkRequest(req, syscall.AF_UNSPEC)
if err != nil {
return nil, err
}
return syscall.ParseNetlinkMessage(data)
}