1
0
mirror of https://github.com/osrg/gobgp.git synced 2024-05-11 05:55:10 +00:00
Files
Joshua DeWald 02fa6129af Implementation of Dynamic Neighbor MD5 (Issue #2589)
Adds usage of the "prefix-based" TCP MD5 for dynamic
neighbors. Non-dynamic neighbors will continue to use
non-prefix based, which makes it more compatible with
running on older kernels, as only 4.14+ includes the
necessary support.

This change also includes tests of dynamic peers in general.
2023-11-07 11:28:12 -08:00

205 lines
4.8 KiB
Go

// Copyright (C) 2016 Nippon Telegraph and Telephone Corporation.
//
// 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.
//go:build linux
// +build linux
package server
import (
"fmt"
"net"
"os"
"strings"
"syscall"
"golang.org/x/sys/unix"
"github.com/osrg/gobgp/v3/pkg/log"
)
const (
ipv6MinHopCount = 73 // Generalized TTL Security Mechanism (RFC5082)
)
func buildTcpMD5Sig(address, key string) *unix.TCPMD5Sig {
t := unix.TCPMD5Sig{}
var addr net.IP
if strings.Contains(address, "/") {
var err error
var ipnet *net.IPNet
addr, ipnet, err = net.ParseCIDR(address)
if err != nil {
return nil
}
prefixlen, _ := ipnet.Mask.Size()
t.Prefixlen = uint8(prefixlen)
t.Flags = unix.TCP_MD5SIG_FLAG_PREFIX
} else {
addr = net.ParseIP(address)
}
if addr.To4() != nil {
t.Addr.Family = unix.AF_INET
copy(t.Addr.Data[2:], addr.To4())
} else {
t.Addr.Family = unix.AF_INET6
copy(t.Addr.Data[6:], addr.To16())
}
t.Keylen = uint16(len(key))
copy(t.Key[0:], []byte(key))
return &t
}
func setTCPMD5SigSockopt(l *net.TCPListener, address string, key string) error {
sc, err := l.SyscallConn()
if err != nil {
return err
}
var sockerr error
t := buildTcpMD5Sig(address, key)
if t == nil {
return fmt.Errorf("unable to generate TcpMD5Sig from %s", address)
}
if err := sc.Control(func(s uintptr) {
opt := unix.TCP_MD5SIG
if t.Prefixlen != 0 {
opt = unix.TCP_MD5SIG_EXT
}
sockerr = unix.SetsockoptTCPMD5Sig(int(s), unix.IPPROTO_TCP, opt, t)
}); err != nil {
return err
}
return sockerr
}
func setBindToDevSockopt(sc syscall.RawConn, device string) error {
return setsockOptString(sc, syscall.SOL_SOCKET, syscall.SO_BINDTODEVICE, device)
}
func setTCPTTLSockopt(conn *net.TCPConn, ttl int) error {
family := extractFamilyFromTCPConn(conn)
sc, err := conn.SyscallConn()
if err != nil {
return err
}
return setsockoptIpTtl(sc, family, ttl)
}
func setTCPMinTTLSockopt(conn *net.TCPConn, ttl int) error {
family := extractFamilyFromTCPConn(conn)
sc, err := conn.SyscallConn()
if err != nil {
return err
}
level := syscall.IPPROTO_IP
name := syscall.IP_MINTTL
if family == syscall.AF_INET6 {
level = syscall.IPPROTO_IPV6
name = ipv6MinHopCount
}
return setsockOptInt(sc, level, name, ttl)
}
func setTCPMSSSockopt(conn *net.TCPConn, mss uint16) error {
family := extractFamilyFromTCPConn(conn)
sc, err := conn.SyscallConn()
if err != nil {
return err
}
return setsockoptTcpMss(sc, family, mss)
}
func dialerControl(logger log.Logger, network, address string, c syscall.RawConn, ttl, minTtl uint8, mss uint16, password string, bindInterface string) error {
family := syscall.AF_INET
raddr, _ := net.ResolveTCPAddr("tcp", address)
if raddr.IP.To4() == nil {
family = syscall.AF_INET6
}
var sockerr error
if password != "" {
addr, _, _ := net.SplitHostPort(address)
t := buildTcpMD5Sig(addr, password)
if err := c.Control(func(fd uintptr) {
sockerr = os.NewSyscallError("setsockopt", unix.SetsockoptTCPMD5Sig(int(fd), unix.IPPROTO_TCP, unix.TCP_MD5SIG, t))
}); err != nil {
return err
}
if sockerr != nil {
return sockerr
}
}
if ttl != 0 {
if err := c.Control(func(fd uintptr) {
level := syscall.IPPROTO_IP
name := syscall.IP_TTL
if family == syscall.AF_INET6 {
level = syscall.IPPROTO_IPV6
name = syscall.IPV6_UNICAST_HOPS
}
sockerr = os.NewSyscallError("setsockopt", syscall.SetsockoptInt(int(fd), level, name, int(ttl)))
}); err != nil {
return err
}
if sockerr != nil {
return sockerr
}
}
if minTtl != 0 {
if err := c.Control(func(fd uintptr) {
level := syscall.IPPROTO_IP
name := syscall.IP_MINTTL
if family == syscall.AF_INET6 {
level = syscall.IPPROTO_IPV6
name = ipv6MinHopCount
}
sockerr = os.NewSyscallError("setsockopt", syscall.SetsockoptInt(int(fd), level, name, int(minTtl)))
}); err != nil {
return err
}
if sockerr != nil {
return sockerr
}
}
if mss != 0 {
if err := c.Control(func(fd uintptr) {
level := syscall.IPPROTO_TCP
name := syscall.TCP_MAXSEG
sockerr = os.NewSyscallError("setsockopt", syscall.SetsockoptInt(int(fd), level, name, int(mss)))
}); err != nil {
return err
}
if sockerr != nil {
return sockerr
}
}
if bindInterface != "" {
if err := setBindToDevSockopt(c, bindInterface); err != nil {
return err
}
}
return nil
}