2020-07-06 18:34:52 +01:00
|
|
|
// Copyright (c) 2020 Tailscale Inc & AUTHORS All rights reserved.
|
|
|
|
// Use of this source code is governed by a BSD-style
|
|
|
|
// license that can be found in the LICENSE file.
|
|
|
|
|
|
|
|
package interfaces
|
|
|
|
|
|
|
|
import (
|
2020-11-19 17:05:32 +00:00
|
|
|
"errors"
|
2021-03-03 05:13:29 +00:00
|
|
|
"fmt"
|
2021-03-15 20:00:20 +00:00
|
|
|
"log"
|
2021-03-03 05:13:29 +00:00
|
|
|
"net"
|
all: convert more code to use net/netip directly
perl -i -npe 's,netaddr.IPPrefixFrom,netip.PrefixFrom,' $(git grep -l -F netaddr.)
perl -i -npe 's,netaddr.IPPortFrom,netip.AddrPortFrom,' $(git grep -l -F netaddr. )
perl -i -npe 's,netaddr.IPPrefix,netip.Prefix,g' $(git grep -l -F netaddr. )
perl -i -npe 's,netaddr.IPPort,netip.AddrPort,g' $(git grep -l -F netaddr. )
perl -i -npe 's,netaddr.IP\b,netip.Addr,g' $(git grep -l -F netaddr. )
perl -i -npe 's,netaddr.IPv6Raw\b,netip.AddrFrom16,g' $(git grep -l -F netaddr. )
goimports -w .
Then delete some stuff from the net/netaddr shim package which is no
longer neeed.
Updates #5162
Change-Id: Ia7a86893fe21c7e3ee1ec823e8aba288d4566cd8
Signed-off-by: Brad Fitzpatrick <bradfitz@tailscale.com>
2022-07-26 05:14:09 +01:00
|
|
|
"net/netip"
|
2021-03-03 05:13:29 +00:00
|
|
|
"syscall"
|
2020-07-06 18:34:52 +01:00
|
|
|
|
2021-03-03 05:13:29 +00:00
|
|
|
"golang.org/x/net/route"
|
2021-03-15 20:00:20 +00:00
|
|
|
"golang.org/x/sys/unix"
|
2022-07-25 04:08:42 +01:00
|
|
|
"tailscale.com/net/netaddr"
|
2020-07-06 18:34:52 +01:00
|
|
|
)
|
|
|
|
|
2021-12-29 18:37:33 +00:00
|
|
|
func defaultRoute() (d DefaultRouteDetails, err error) {
|
2021-03-03 05:13:29 +00:00
|
|
|
idx, err := DefaultRouteInterfaceIndex()
|
|
|
|
if err != nil {
|
2021-12-29 18:37:33 +00:00
|
|
|
return d, err
|
2021-03-03 05:13:29 +00:00
|
|
|
}
|
|
|
|
iface, err := net.InterfaceByIndex(idx)
|
|
|
|
if err != nil {
|
2021-12-29 18:37:33 +00:00
|
|
|
return d, err
|
2021-03-03 05:13:29 +00:00
|
|
|
}
|
2021-12-29 18:37:33 +00:00
|
|
|
d.InterfaceName = iface.Name
|
|
|
|
d.InterfaceIndex = idx
|
|
|
|
return d, nil
|
2021-03-03 05:13:29 +00:00
|
|
|
}
|
|
|
|
|
2021-05-10 16:23:41 +01:00
|
|
|
// fetchRoutingTable calls route.FetchRIB, fetching NET_RT_DUMP2.
|
2021-04-23 21:23:19 +01:00
|
|
|
func fetchRoutingTable() (rib []byte, err error) {
|
2021-05-10 16:23:41 +01:00
|
|
|
return route.FetchRIB(syscall.AF_UNSPEC, syscall.NET_RT_DUMP2, 0)
|
2021-04-23 21:23:19 +01:00
|
|
|
}
|
|
|
|
|
2021-03-03 05:13:29 +00:00
|
|
|
func DefaultRouteInterfaceIndex() (int, error) {
|
|
|
|
// $ netstat -nr
|
|
|
|
// Routing tables
|
|
|
|
// Internet:
|
|
|
|
// Destination Gateway Flags Netif Expire
|
|
|
|
// default 10.0.0.1 UGSc en0 <-- want this one
|
|
|
|
// default 10.0.0.1 UGScI en1
|
|
|
|
|
|
|
|
// From man netstat:
|
|
|
|
// U RTF_UP Route usable
|
|
|
|
// G RTF_GATEWAY Destination requires forwarding by intermediary
|
|
|
|
// S RTF_STATIC Manually added
|
|
|
|
// c RTF_PRCLONING Protocol-specified generate new routes on use
|
|
|
|
// I RTF_IFSCOPE Route is associated with an interface scope
|
|
|
|
|
2021-04-23 21:23:19 +01:00
|
|
|
rib, err := fetchRoutingTable()
|
2021-03-03 05:13:29 +00:00
|
|
|
if err != nil {
|
|
|
|
return 0, fmt.Errorf("route.FetchRIB: %w", err)
|
|
|
|
}
|
|
|
|
msgs, err := route.ParseRIB(syscall.NET_RT_IFLIST2, rib)
|
|
|
|
if err != nil {
|
|
|
|
return 0, fmt.Errorf("route.ParseRIB: %w", err)
|
|
|
|
}
|
|
|
|
indexSeen := map[int]int{} // index => count
|
|
|
|
for _, m := range msgs {
|
|
|
|
rm, ok := m.(*route.RouteMessage)
|
|
|
|
if !ok {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
const RTF_GATEWAY = 0x2
|
|
|
|
const RTF_IFSCOPE = 0x1000000
|
|
|
|
if rm.Flags&RTF_GATEWAY == 0 {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if rm.Flags&RTF_IFSCOPE != 0 {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
indexSeen[rm.Index]++
|
|
|
|
}
|
|
|
|
if len(indexSeen) == 0 {
|
|
|
|
return 0, errors.New("no gateway index found")
|
|
|
|
}
|
|
|
|
if len(indexSeen) == 1 {
|
|
|
|
for idx := range indexSeen {
|
|
|
|
return idx, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0, fmt.Errorf("ambiguous gateway interfaces found: %v", indexSeen)
|
|
|
|
}
|
2021-03-15 20:00:20 +00:00
|
|
|
|
|
|
|
func init() {
|
|
|
|
likelyHomeRouterIP = likelyHomeRouterIPDarwinFetchRIB
|
|
|
|
}
|
|
|
|
|
all: convert more code to use net/netip directly
perl -i -npe 's,netaddr.IPPrefixFrom,netip.PrefixFrom,' $(git grep -l -F netaddr.)
perl -i -npe 's,netaddr.IPPortFrom,netip.AddrPortFrom,' $(git grep -l -F netaddr. )
perl -i -npe 's,netaddr.IPPrefix,netip.Prefix,g' $(git grep -l -F netaddr. )
perl -i -npe 's,netaddr.IPPort,netip.AddrPort,g' $(git grep -l -F netaddr. )
perl -i -npe 's,netaddr.IP\b,netip.Addr,g' $(git grep -l -F netaddr. )
perl -i -npe 's,netaddr.IPv6Raw\b,netip.AddrFrom16,g' $(git grep -l -F netaddr. )
goimports -w .
Then delete some stuff from the net/netaddr shim package which is no
longer neeed.
Updates #5162
Change-Id: Ia7a86893fe21c7e3ee1ec823e8aba288d4566cd8
Signed-off-by: Brad Fitzpatrick <bradfitz@tailscale.com>
2022-07-26 05:14:09 +01:00
|
|
|
func likelyHomeRouterIPDarwinFetchRIB() (ret netip.Addr, ok bool) {
|
2021-04-23 21:23:19 +01:00
|
|
|
rib, err := fetchRoutingTable()
|
2021-03-15 20:00:20 +00:00
|
|
|
if err != nil {
|
|
|
|
log.Printf("routerIP/FetchRIB: %v", err)
|
|
|
|
return ret, false
|
|
|
|
}
|
|
|
|
msgs, err := route.ParseRIB(syscall.NET_RT_IFLIST2, rib)
|
|
|
|
if err != nil {
|
|
|
|
log.Printf("routerIP/ParseRIB: %v", err)
|
|
|
|
return ret, false
|
|
|
|
}
|
|
|
|
for _, m := range msgs {
|
|
|
|
rm, ok := m.(*route.RouteMessage)
|
|
|
|
if !ok {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
const RTF_GATEWAY = 0x2
|
|
|
|
const RTF_IFSCOPE = 0x1000000
|
|
|
|
if rm.Flags&RTF_GATEWAY == 0 {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if rm.Flags&RTF_IFSCOPE != 0 {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if len(rm.Addrs) > unix.RTAX_GATEWAY {
|
|
|
|
dst4, ok := rm.Addrs[unix.RTAX_DST].(*route.Inet4Addr)
|
|
|
|
if !ok || dst4.IP != ([4]byte{0, 0, 0, 0}) {
|
|
|
|
// Expect 0.0.0.0 as DST field.
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
gw, ok := rm.Addrs[unix.RTAX_GATEWAY].(*route.Inet4Addr)
|
|
|
|
if !ok {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
return netaddr.IPv4(gw.IP[0], gw.IP[1], gw.IP[2], gw.IP[3]), true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret, false
|
|
|
|
}
|