2023-01-27 13:37:20 -08:00
|
|
|
// Copyright (c) Tailscale Inc & AUTHORS
|
|
|
|
// SPDX-License-Identifier: BSD-3-Clause
|
2020-05-29 00:43:15 +00:00
|
|
|
|
2021-08-05 15:42:39 -07:00
|
|
|
//go:build linux && !android
|
2021-06-24 12:50:47 -07:00
|
|
|
|
2020-05-29 00:43:15 +00:00
|
|
|
package netns
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
2021-06-29 21:13:00 -07:00
|
|
|
"net"
|
2020-05-29 21:58:31 -07:00
|
|
|
"os"
|
2020-05-31 04:31:01 -04:00
|
|
|
"sync"
|
2020-05-29 00:43:15 +00:00
|
|
|
"syscall"
|
|
|
|
|
|
|
|
"golang.org/x/sys/unix"
|
2022-09-10 17:46:09 -07:00
|
|
|
"tailscale.com/envknob"
|
2023-04-17 16:01:41 -07:00
|
|
|
"tailscale.com/net/netmon"
|
2021-11-18 12:18:02 -08:00
|
|
|
"tailscale.com/types/logger"
|
2023-06-16 18:54:58 +00:00
|
|
|
"tailscale.com/util/linuxfw"
|
2020-05-29 00:43:15 +00:00
|
|
|
)
|
|
|
|
|
2021-06-29 21:13:00 -07:00
|
|
|
// socketMarkWorksOnce is the sync.Once & cached value for useSocketMark.
|
|
|
|
var socketMarkWorksOnce struct {
|
2020-05-31 14:01:20 -07:00
|
|
|
sync.Once
|
|
|
|
v bool
|
|
|
|
}
|
2020-05-31 04:31:01 -04:00
|
|
|
|
2021-06-29 21:13:00 -07:00
|
|
|
// socketMarkWorks returns whether SO_MARK works.
|
|
|
|
func socketMarkWorks() bool {
|
|
|
|
addr, err := net.ResolveUDPAddr("udp", "127.0.0.1:1")
|
|
|
|
if err != nil {
|
|
|
|
return true // unsure, returning true does the least harm.
|
|
|
|
}
|
|
|
|
|
|
|
|
sConn, err := net.DialUDP("udp", nil, addr)
|
|
|
|
if err != nil {
|
|
|
|
return true // unsure, return true
|
|
|
|
}
|
|
|
|
defer sConn.Close()
|
|
|
|
|
|
|
|
rConn, err := sConn.SyscallConn()
|
|
|
|
if err != nil {
|
|
|
|
return true // unsure, return true
|
|
|
|
}
|
|
|
|
|
|
|
|
var sockErr error
|
|
|
|
err = rConn.Control(func(fd uintptr) {
|
|
|
|
sockErr = setBypassMark(fd)
|
|
|
|
})
|
|
|
|
if err != nil || sockErr != nil {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
2022-09-14 12:49:39 -07:00
|
|
|
var forceBindToDevice = envknob.RegisterBool("TS_FORCE_LINUX_BIND_TO_DEVICE")
|
2022-09-10 17:46:09 -07:00
|
|
|
|
2022-09-12 14:31:20 -07:00
|
|
|
// UseSocketMark reports whether SO_MARK is in use.
|
2020-05-31 04:31:01 -04:00
|
|
|
// If it doesn't, we have to use SO_BINDTODEVICE on our sockets instead.
|
2022-09-12 14:31:20 -07:00
|
|
|
func UseSocketMark() bool {
|
2022-09-14 12:49:39 -07:00
|
|
|
if forceBindToDevice() {
|
2022-09-10 17:46:09 -07:00
|
|
|
return false
|
|
|
|
}
|
2021-06-29 21:13:00 -07:00
|
|
|
socketMarkWorksOnce.Do(func() {
|
2022-01-11 15:26:58 -08:00
|
|
|
socketMarkWorksOnce.v = socketMarkWorks()
|
2020-05-31 14:01:20 -07:00
|
|
|
})
|
2021-06-29 21:13:00 -07:00
|
|
|
return socketMarkWorksOnce.v
|
2020-05-31 14:01:20 -07:00
|
|
|
}
|
2020-05-31 04:31:01 -04:00
|
|
|
|
|
|
|
// ignoreErrors returns true if we should ignore setsocketopt errors in
|
|
|
|
// this instance.
|
|
|
|
func ignoreErrors() bool {
|
|
|
|
if os.Getuid() != 0 {
|
|
|
|
// only root can manipulate these socket flags
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2023-04-17 16:01:41 -07:00
|
|
|
func control(logger.Logf, *netmon.Monitor) func(network, address string, c syscall.RawConn) error {
|
2021-11-18 12:18:02 -08:00
|
|
|
return controlC
|
|
|
|
}
|
|
|
|
|
|
|
|
// controlC marks c as necessary to dial in a separate network namespace.
|
2020-05-29 00:43:15 +00:00
|
|
|
//
|
|
|
|
// It's intentionally the same signature as net.Dialer.Control
|
|
|
|
// and net.ListenConfig.Control.
|
2021-11-18 12:18:02 -08:00
|
|
|
func controlC(network, address string, c syscall.RawConn) error {
|
2021-06-29 21:13:00 -07:00
|
|
|
if isLocalhost(address) {
|
2021-07-09 04:40:41 -07:00
|
|
|
// Don't bind to an interface for localhost connections.
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-05-31 14:29:54 -07:00
|
|
|
var sockErr error
|
|
|
|
err := c.Control(func(fd uintptr) {
|
2022-09-12 14:31:20 -07:00
|
|
|
if UseSocketMark() {
|
2020-05-31 14:29:54 -07:00
|
|
|
sockErr = setBypassMark(fd)
|
|
|
|
} else {
|
|
|
|
sockErr = bindToDevice(fd)
|
2020-05-29 21:58:31 -07:00
|
|
|
}
|
2020-05-31 14:29:54 -07:00
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("RawConn.Control on %T: %w", c, err)
|
|
|
|
}
|
|
|
|
if sockErr != nil && ignoreErrors() {
|
|
|
|
// TODO(bradfitz): maybe log once? probably too spammy for e.g. CLI tools like tailscale netcheck.
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return sockErr
|
|
|
|
}
|
|
|
|
|
|
|
|
func setBypassMark(fd uintptr) error {
|
2023-06-16 18:54:58 +00:00
|
|
|
if err := unix.SetsockoptInt(int(fd), unix.SOL_SOCKET, unix.SO_MARK, linuxfw.TailscaleBypassMarkNum); err != nil {
|
2020-05-31 14:29:54 -07:00
|
|
|
return fmt.Errorf("setting SO_MARK bypass: %w", err)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func bindToDevice(fd uintptr) error {
|
2024-04-27 21:18:18 -07:00
|
|
|
ifc, err := netmon.DefaultRouteInterface()
|
2020-05-31 14:29:54 -07:00
|
|
|
if err != nil {
|
|
|
|
// Make sure we bind to *some* interface,
|
|
|
|
// or we could get a routing loop.
|
|
|
|
// "lo" is always wrong, but if we don't have
|
|
|
|
// a default route anyway, it doesn't matter.
|
|
|
|
ifc = "lo"
|
|
|
|
}
|
|
|
|
if err := unix.SetsockoptString(int(fd), unix.SOL_SOCKET, unix.SO_BINDTODEVICE, ifc); err != nil {
|
|
|
|
return fmt.Errorf("setting SO_BINDTODEVICE: %w", err)
|
2020-05-29 00:43:15 +00:00
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|