summaryrefslogtreecommitdiffhomepage
path: root/pkg/sentry/socket
diff options
context:
space:
mode:
authorMikael Magnusson <mikma@users.sourceforge.net>2021-02-08 15:47:02 +0100
committerMikael Magnusson <mikma@users.sourceforge.net>2021-09-24 18:54:58 +0000
commit96667cf4a3c30726cb4242f749afcee42ce7cd43 (patch)
treea2518b5e8aa07ca3396a8967a9d58081309808eb /pkg/sentry/socket
parent12175748aba75c0b3be5b3981763c1a1f5e73763 (diff)
netlink: Implement neighbor messages
Implement dumpNeighbors, newNeigh and delNeigh. Fixes #5744
Diffstat (limited to 'pkg/sentry/socket')
-rw-r--r--pkg/sentry/socket/hostinet/stack.go15
-rw-r--r--pkg/sentry/socket/netlink/route/protocol.go130
-rw-r--r--pkg/sentry/socket/netstack/stack.go93
3 files changed, 238 insertions, 0 deletions
diff --git a/pkg/sentry/socket/hostinet/stack.go b/pkg/sentry/socket/hostinet/stack.go
index 61111ac6c..b80d64c9a 100644
--- a/pkg/sentry/socket/hostinet/stack.go
+++ b/pkg/sentry/socket/hostinet/stack.go
@@ -333,6 +333,21 @@ func (*Stack) RemoveInterfaceAddr(int32, inet.InterfaceAddr) error {
return linuxerr.EACCES
}
+// Neighbors implements inet.Stack.Neighbors.
+func (s *Stack) Neighbors() ([]inet.Neighbor, error) {
+ return nil, linuxerr.EACCES
+}
+
+// AddNeighbor implements inet.Stack.AddNeighbor.
+func (s *Stack) AddNeighbor(inet.Neighbor) error {
+ return linuxerr.EACCES
+}
+
+// RemoveNeighbor implements inet.Stack.RemoveNeighbor.
+func (s *Stack) RemoveNeighbor(inet.Neighbor) error {
+ return linuxerr.EACCES
+}
+
// SupportsIPv6 implements inet.Stack.SupportsIPv6.
func (s *Stack) SupportsIPv6() bool {
return s.supportsIPv6
diff --git a/pkg/sentry/socket/netlink/route/protocol.go b/pkg/sentry/socket/netlink/route/protocol.go
index d526acb73..38f714c48 100644
--- a/pkg/sentry/socket/netlink/route/protocol.go
+++ b/pkg/sentry/socket/netlink/route/protocol.go
@@ -275,6 +275,130 @@ func (p *Protocol) dumpAddrs(ctx context.Context, msg *netlink.Message, ms *netl
return nil
}
+// dumpNeighbors handles RTM_GETNEIGH dump requests.
+func (p *Protocol) dumpNeighbors(ctx context.Context, msg *netlink.Message, ms *netlink.MessageSet) *syserr.Error {
+ // RTM_GETNEIGH dump requests need not contain anything more than the
+ // netlink header and 1 byte protocol family common to all
+ // NETLINK_ROUTE requests.
+
+ // The RTM_GETNEIGH dump response is a set of RTM_NEWNEIGH messages each
+ // containing an NeighborMessage followed by a set of netlink
+ // attributes.
+
+ // We always send back an NLMSG_DONE.
+ ms.Multi = true
+
+ stack := inet.StackFromContext(ctx)
+ if stack == nil {
+ // No network devices.
+ return nil
+ }
+
+ nas, err := stack.Neighbors(); if err != nil {
+ return syserr.FromError(err)
+ }
+
+ for _, na := range nas {
+ m := ms.AddMessage(linux.NetlinkMessageHeader{
+ Type: linux.RTM_NEWNEIGH,
+ })
+
+ m.Put(&linux.NeighborMessage{
+ Family: na.Family,
+ IfIndex: na.Idx,
+ State: na.State,
+ })
+
+ m.PutAttr(linux.NDA_DST, primitive.AsByteSlice(na.Addr))
+ if len(na.LinkAddr) > 0 {
+ m.PutAttr(linux.NDA_LLADDR, primitive.AsByteSlice(na.LinkAddr))
+ }
+ }
+
+ return nil
+}
+
+// newNeigh handles RTM_NEWNEIGH requests.
+func (p *Protocol) newNeigh(ctx context.Context, msg *netlink.Message, ms *netlink.MessageSet) *syserr.Error {
+ stack := inet.StackFromContext(ctx)
+ if stack == nil {
+ // No network stack.
+ return syserr.ErrProtocolNotSupported
+ }
+
+ var nm linux.NeighborMessage
+ attrs, ok := msg.GetData(&nm)
+ if !ok {
+ return syserr.ErrInvalidArgument
+ }
+
+ var addr []byte
+ var llAddr []byte
+
+ for !attrs.Empty() {
+ ahdr, value, rest, ok := attrs.ParseFirst()
+ if !ok {
+ return syserr.ErrInvalidArgument
+ }
+ attrs = rest
+
+ switch ahdr.Type {
+ case linux.NDA_DST:
+ addr = value
+ case linux.NDA_LLADDR:
+ llAddr = value
+ default:
+ return syserr.ErrNotSupported
+ }
+ }
+
+ return syserr.FromError(stack.AddNeighbor(inet.Neighbor{
+ Family: nm.Family,
+ Idx: nm.IfIndex,
+ Addr: addr,
+ State: nm.State,
+ LinkAddr: llAddr,
+ }))
+}
+
+// delNeigh handles RTM_DELNEIGH requests.
+func (p *Protocol) delNeigh(ctx context.Context, msg *netlink.Message, ms *netlink.MessageSet) *syserr.Error {
+ stack := inet.StackFromContext(ctx)
+ if stack == nil {
+ // No network stack.
+ return syserr.ErrProtocolNotSupported
+ }
+
+ var nm linux.NeighborMessage
+ attrs, ok := msg.GetData(&nm)
+ if !ok {
+ return syserr.ErrInvalidArgument
+ }
+
+ var addr []byte
+
+ for !attrs.Empty() {
+ ahdr, value, rest, ok := attrs.ParseFirst()
+ if !ok {
+ return syserr.ErrInvalidArgument
+ }
+ attrs = rest
+
+ switch ahdr.Type {
+ case linux.NDA_DST:
+ addr = value
+ default:
+ return syserr.ErrNotSupported
+ }
+ }
+
+ return syserr.FromError(stack.RemoveNeighbor(inet.Neighbor{
+ Family: nm.Family,
+ Idx: nm.IfIndex,
+ Addr: addr,
+ }))
+}
+
// commonPrefixLen reports the length of the longest IP address prefix.
// This is a simplied version from Golang's src/net/addrselect.go.
func commonPrefixLen(a, b []byte) (cpl int) {
@@ -571,6 +695,8 @@ func (p *Protocol) ProcessMessage(ctx context.Context, msg *netlink.Message, ms
return p.dumpAddrs(ctx, msg, ms)
case linux.RTM_GETROUTE:
return p.dumpRoutes(ctx, msg, ms)
+ case linux.RTM_GETNEIGH:
+ return p.dumpNeighbors(ctx, msg, ms)
default:
return syserr.ErrNotSupported
}
@@ -586,6 +712,10 @@ func (p *Protocol) ProcessMessage(ctx context.Context, msg *netlink.Message, ms
return p.newAddr(ctx, msg, ms)
case linux.RTM_DELADDR:
return p.delAddr(ctx, msg, ms)
+ case linux.RTM_NEWNEIGH:
+ return p.newNeigh(ctx, msg, ms)
+ case linux.RTM_DELNEIGH:
+ return p.delNeigh(ctx, msg, ms)
default:
return syserr.ErrNotSupported
}
diff --git a/pkg/sentry/socket/netstack/stack.go b/pkg/sentry/socket/netstack/stack.go
index ea199f223..3fceb4cd5 100644
--- a/pkg/sentry/socket/netstack/stack.go
+++ b/pkg/sentry/socket/netstack/stack.go
@@ -204,6 +204,99 @@ func (s *Stack) RemoveInterfaceAddr(idx int32, addr inet.InterfaceAddr) error {
return nil
}
+func (s *Stack) appendNeighbor(ns []inet.Neighbor, id tcpip.NICID, family uint8, protocol tcpip.NetworkProtocolNumber) ([]inet.Neighbor, error) {
+ ne, err := s.Stack.Neighbors(id, protocol); if _, ok := err.(*tcpip.ErrNotSupported); ok {
+ return ns, nil
+ } else if err != nil {
+ return ns, syserr.TranslateNetstackError(err).ToError()
+ }
+
+ for _, ni := range ne {
+ var State uint16 = 0
+
+ switch ni.State {
+ case stack.Unknown:
+ State = linux.NUD_NONE
+ case stack.Incomplete:
+ State = linux.NUD_INCOMPLETE
+ case stack.Reachable:
+ State = linux.NUD_REACHABLE
+ case stack.Stale:
+ State = linux.NUD_STALE
+ case stack.Delay:
+ State = linux.NUD_DELAY
+ case stack.Probe:
+ State = linux.NUD_PROBE
+ case stack.Static:
+ State = linux.NUD_PERMANENT
+ case stack.Unreachable:
+ State = linux.NUD_FAILED
+ }
+
+ ns = append(ns, inet.Neighbor{
+ Family: family,
+ Idx: int32(id),
+ Addr: []byte(ni.Addr),
+ LinkAddr: []byte(ni.LinkAddr),
+ State: State,
+ })
+ }
+
+ return ns, nil
+}
+
+// Neighbors implements inet.Stack.Neighbors.
+func (s *Stack) Neighbors() ([]inet.Neighbor, error) {
+ var err error
+ ns := []inet.Neighbor{}
+ for id, _ := range s.Stack.NICInfo() {
+ ns, err = s.appendNeighbor(ns, id, linux.AF_INET, ipv4.ProtocolNumber); if err != nil {
+ return nil, err
+ }
+ ns, err = s.appendNeighbor(ns, id, linux.AF_INET6, ipv6.ProtocolNumber); if err != nil {
+ return nil, err
+ }
+ }
+
+ return ns, nil
+}
+
+func (s *Stack) AddNeighbor(n inet.Neighbor) error {
+ var protocol tcpip.NetworkProtocolNumber
+
+ switch n.Family {
+ case linux.AF_INET:
+ protocol = ipv4.ProtocolNumber
+ case linux.AF_INET6:
+ protocol = ipv6.ProtocolNumber
+ default:
+ panic(fmt.Sprintf("AddStaticNeighbor(%v) failed: unsupported family", n.Family))
+ }
+
+ if n.State != linux.NUD_PERMANENT {
+ return syserr.ErrInvalidArgument.ToError()
+ }
+
+ err := s.Stack.AddStaticNeighbor(tcpip.NICID(n.Idx), protocol, tcpip.Address(n.Addr), tcpip.LinkAddress(n.LinkAddr))
+ return syserr.TranslateNetstackError(err).ToError()
+}
+
+func (s *Stack) RemoveNeighbor(n inet.Neighbor) error {
+ var protocol tcpip.NetworkProtocolNumber
+
+ switch n.Family {
+ case linux.AF_INET:
+ protocol = ipv4.ProtocolNumber
+ case linux.AF_INET6:
+ protocol = ipv6.ProtocolNumber
+ default:
+ panic(fmt.Sprintf("RemoveNeighbor(%v) failed: unsupported family", n.Family))
+ }
+
+ err := s.Stack.RemoveNeighbor(tcpip.NICID(n.Idx), protocol, tcpip.Address(n.Addr))
+ return syserr.TranslateNetstackError(err).ToError()
+}
+
// TCPReceiveBufferSize implements inet.Stack.TCPReceiveBufferSize.
func (s *Stack) TCPReceiveBufferSize() (inet.TCPBufferSize, error) {
var rs tcpip.TCPReceiveBufferSizeRangeOption