| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556 |
- // 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 wgengine
- import (
- "bufio"
- "bytes"
- crand "crypto/rand"
- "errors"
- "fmt"
- "io"
- "os"
- "reflect"
- "runtime"
- "strconv"
- "strings"
- "sync"
- "sync/atomic"
- "time"
- "go4.org/mem"
- "golang.zx2c4.com/wireguard/device"
- "golang.zx2c4.com/wireguard/tun"
- "inet.af/netaddr"
- "tailscale.com/control/controlclient"
- "tailscale.com/health"
- "tailscale.com/ipn/ipnstate"
- "tailscale.com/net/dns"
- "tailscale.com/net/dns/resolver"
- "tailscale.com/net/flowtrack"
- "tailscale.com/net/interfaces"
- "tailscale.com/net/packet"
- "tailscale.com/net/tsaddr"
- "tailscale.com/net/tshttpproxy"
- "tailscale.com/net/tstun"
- "tailscale.com/tailcfg"
- "tailscale.com/tstime/mono"
- "tailscale.com/types/dnstype"
- "tailscale.com/types/ipproto"
- "tailscale.com/types/key"
- "tailscale.com/types/logger"
- "tailscale.com/types/netmap"
- "tailscale.com/util/deephash"
- "tailscale.com/version"
- "tailscale.com/wgengine/filter"
- "tailscale.com/wgengine/magicsock"
- "tailscale.com/wgengine/monitor"
- "tailscale.com/wgengine/router"
- "tailscale.com/wgengine/wgcfg"
- "tailscale.com/wgengine/wglog"
- )
- const magicDNSPort = 53
- var magicDNSIP = netaddr.IPv4(100, 100, 100, 100)
- // Lazy wireguard-go configuration parameters.
- const (
- // lazyPeerIdleThreshold is the idle duration after
- // which we remove a peer from the wireguard configuration.
- // (This includes peers that have never been idle, which
- // effectively have infinite idleness)
- lazyPeerIdleThreshold = 5 * time.Minute
- // packetSendTimeUpdateFrequency controls how often we record
- // the time that we wrote a packet to an IP address.
- packetSendTimeUpdateFrequency = 10 * time.Second
- // packetSendRecheckWireguardThreshold controls how long we can go
- // between packet sends to an IP before checking to see
- // whether this IP address needs to be added back to the
- // Wireguard peer oconfig.
- packetSendRecheckWireguardThreshold = 1 * time.Minute
- )
- // statusPollInterval is how often we ask wireguard-go for its engine
- // status (as long as there's activity). See docs on its use below.
- const statusPollInterval = 1 * time.Minute
- type userspaceEngine struct {
- logf logger.Logf
- wgLogger *wglog.Logger //a wireguard-go logging wrapper
- reqCh chan struct{}
- waitCh chan struct{} // chan is closed when first Close call completes; contrast with closing bool
- timeNow func() mono.Time
- tundev *tstun.Wrapper
- wgdev *device.Device
- router router.Router
- confListenPort uint16 // original conf.ListenPort
- dns *dns.Manager
- magicConn *magicsock.Conn
- linkMon *monitor.Mon
- linkMonOwned bool // whether we created linkMon (and thus need to close it)
- linkMonUnregister func() // unsubscribes from changes; used regardless of linkMonOwned
- birdClient BIRDClient // or nil
- testMaybeReconfigHook func() // for tests; if non-nil, fires if maybeReconfigWireguardLocked called
- // isLocalAddr reports the whether an IP is assigned to the local
- // tunnel interface. It's used to reflect local packets
- // incorrectly sent to us.
- isLocalAddr atomic.Value // of func(netaddr.IP)bool
- // isDNSIPOverTailscale reports the whether a DNS resolver's IP
- // is being routed over Tailscale.
- isDNSIPOverTailscale atomic.Value // of func(netaddr.IP)bool
- wgLock sync.Mutex // serializes all wgdev operations; see lock order comment below
- lastCfgFull wgcfg.Config
- lastNMinPeers int
- lastRouterSig deephash.Sum // of router.Config
- lastEngineSigFull deephash.Sum // of full wireguard config
- lastEngineSigTrim deephash.Sum // of trimmed wireguard config
- lastDNSConfig *dns.Config
- recvActivityAt map[key.NodePublic]mono.Time
- trimmedNodes map[key.NodePublic]bool // set of node keys of peers currently excluded from wireguard config
- sentActivityAt map[netaddr.IP]*mono.Time // value is accessed atomically
- destIPActivityFuncs map[netaddr.IP]func()
- statusBufioReader *bufio.Reader // reusable for UAPI
- lastStatusPollTime mono.Time // last time we polled the engine status
- lastIsSubnetRouter bool // was the node a primary subnet router in the last run.
- mu sync.Mutex // guards following; see lock order comment below
- netMap *netmap.NetworkMap // or nil
- closing bool // Close was called (even if we're still closing)
- statusCallback StatusCallback
- peerSequence []key.NodePublic
- endpoints []tailcfg.Endpoint
- pendOpen map[flowtrack.Tuple]*pendingOpenFlow // see pendopen.go
- networkMapCallbacks map[*someHandle]NetworkMapCallback
- tsIPByIPPort map[netaddr.IPPort]netaddr.IP // allows registration of IP:ports as belonging to a certain Tailscale IP for whois lookups
- pongCallback map[[8]byte]func(packet.TSMPPongReply) // for TSMP pong responses
- // Lock ordering: magicsock.Conn.mu, wgLock, then mu.
- }
- // InternalsGetter is implemented by Engines that can export their internals.
- type InternalsGetter interface {
- GetInternals() (_ *tstun.Wrapper, _ *magicsock.Conn, ok bool)
- }
- func (e *userspaceEngine) GetInternals() (_ *tstun.Wrapper, _ *magicsock.Conn, ok bool) {
- return e.tundev, e.magicConn, true
- }
- // BIRDClient handles communication with the BIRD Internet Routing Daemon.
- type BIRDClient interface {
- EnableProtocol(proto string) error
- DisableProtocol(proto string) error
- Close() error
- }
- // Config is the engine configuration.
- type Config struct {
- // Tun is the device used by the Engine to exchange packets with
- // the OS.
- // If nil, a fake Device that does nothing is used.
- Tun tun.Device
- // IsTAP is whether Tun is actually a TAP (Layer 2) device that'll
- // require ethernet headers.
- IsTAP bool
- // Router interfaces the Engine to the OS network stack.
- // If nil, a fake Router that does nothing is used.
- Router router.Router
- // DNS interfaces the Engine to the OS DNS resolver configuration.
- // If nil, a fake OSConfigurator that does nothing is used.
- DNS dns.OSConfigurator
- // LinkMonitor optionally provides an existing link monitor to re-use.
- // If nil, a new link monitor is created.
- LinkMonitor *monitor.Mon
- // ListenPort is the port on which the engine will listen.
- // If zero, a port is automatically selected.
- ListenPort uint16
- // RespondToPing determines whether this engine should internally
- // reply to ICMP pings, without involving the OS.
- // Used in "fake" mode for development.
- RespondToPing bool
- // BIRDClient, if non-nil, will be used to configure BIRD whenever
- // this node is a primary subnet router.
- BIRDClient BIRDClient
- }
- func NewFakeUserspaceEngine(logf logger.Logf, listenPort uint16) (Engine, error) {
- logf("Starting userspace wireguard engine (with fake TUN device)")
- return NewUserspaceEngine(logf, Config{
- ListenPort: listenPort,
- RespondToPing: true,
- })
- }
- // NetstackRouterType is a gross cross-package init-time registration
- // from netstack to here, informing this package of netstack's router
- // type.
- var NetstackRouterType reflect.Type
- // IsNetstackRouter reports whether e is either fully netstack based
- // (without TUN) or is at least using netstack for routing.
- func IsNetstackRouter(e Engine) bool {
- switch e := e.(type) {
- case *userspaceEngine:
- if reflect.TypeOf(e.router) == NetstackRouterType {
- return true
- }
- case *watchdogEngine:
- return IsNetstackRouter(e.wrap)
- }
- return IsNetstack(e)
- }
- // IsNetstack reports whether e is a netstack-based TUN-free engine.
- func IsNetstack(e Engine) bool {
- ig, ok := e.(InternalsGetter)
- if !ok {
- return false
- }
- tw, _, ok := ig.GetInternals()
- if !ok {
- return false
- }
- name, err := tw.Name()
- return err == nil && name == "FakeTUN"
- }
- // NewUserspaceEngine creates the named tun device and returns a
- // Tailscale Engine running on it.
- func NewUserspaceEngine(logf logger.Logf, conf Config) (_ Engine, reterr error) {
- var closePool closeOnErrorPool
- defer closePool.closeAllIfError(&reterr)
- if conf.Tun == nil {
- logf("[v1] using fake (no-op) tun device")
- conf.Tun = tstun.NewFake()
- }
- if conf.Router == nil {
- logf("[v1] using fake (no-op) OS network configurator")
- conf.Router = router.NewFake(logf)
- }
- if conf.DNS == nil {
- logf("[v1] using fake (no-op) DNS configurator")
- d, err := dns.NewNoopManager()
- if err != nil {
- return nil, err
- }
- conf.DNS = d
- }
- var tsTUNDev *tstun.Wrapper
- if conf.IsTAP {
- tsTUNDev = tstun.WrapTAP(logf, conf.Tun)
- } else {
- tsTUNDev = tstun.Wrap(logf, conf.Tun)
- }
- closePool.add(tsTUNDev)
- e := &userspaceEngine{
- timeNow: mono.Now,
- logf: logf,
- reqCh: make(chan struct{}, 1),
- waitCh: make(chan struct{}),
- tundev: tsTUNDev,
- router: conf.Router,
- confListenPort: conf.ListenPort,
- birdClient: conf.BIRDClient,
- }
- if e.birdClient != nil {
- // Disable the protocol at start time.
- if err := e.birdClient.DisableProtocol("tailscale"); err != nil {
- return nil, err
- }
- }
- e.isLocalAddr.Store(tsaddr.NewContainsIPFunc(nil))
- e.isDNSIPOverTailscale.Store(tsaddr.NewContainsIPFunc(nil))
- if conf.LinkMonitor != nil {
- e.linkMon = conf.LinkMonitor
- } else {
- mon, err := monitor.New(logf)
- if err != nil {
- return nil, err
- }
- closePool.add(mon)
- e.linkMon = mon
- e.linkMonOwned = true
- }
- tunName, _ := conf.Tun.Name()
- e.dns = dns.NewManager(logf, conf.DNS, e.linkMon, fwdDNSLinkSelector{e, tunName})
- logf("link state: %+v", e.linkMon.InterfaceState())
- unregisterMonWatch := e.linkMon.RegisterChangeCallback(func(changed bool, st *interfaces.State) {
- tshttpproxy.InvalidateCache()
- e.linkChange(changed, st)
- })
- closePool.addFunc(unregisterMonWatch)
- e.linkMonUnregister = unregisterMonWatch
- endpointsFn := func(endpoints []tailcfg.Endpoint) {
- e.mu.Lock()
- e.endpoints = append(e.endpoints[:0], endpoints...)
- e.mu.Unlock()
- e.RequestStatus()
- }
- magicsockOpts := magicsock.Options{
- Logf: logf,
- Port: conf.ListenPort,
- EndpointsFunc: endpointsFn,
- DERPActiveFunc: e.RequestStatus,
- IdleFunc: e.tundev.IdleDuration,
- NoteRecvActivity: e.noteRecvActivity,
- LinkMonitor: e.linkMon,
- }
- var err error
- e.magicConn, err = magicsock.NewConn(magicsockOpts)
- if err != nil {
- return nil, fmt.Errorf("wgengine: %v", err)
- }
- closePool.add(e.magicConn)
- e.magicConn.SetNetworkUp(e.linkMon.InterfaceState().AnyInterfaceUp())
- tsTUNDev.SetDiscoKey(e.magicConn.DiscoPublicKey())
- if conf.RespondToPing {
- e.tundev.PostFilterIn = echoRespondToAll
- }
- e.tundev.PreFilterOut = e.handleLocalPackets
- if debugConnectFailures() {
- if e.tundev.PreFilterIn != nil {
- return nil, errors.New("unexpected PreFilterIn already set")
- }
- e.tundev.PreFilterIn = e.trackOpenPreFilterIn
- if e.tundev.PostFilterOut != nil {
- return nil, errors.New("unexpected PostFilterOut already set")
- }
- e.tundev.PostFilterOut = e.trackOpenPostFilterOut
- }
- e.wgLogger = wglog.NewLogger(logf)
- e.tundev.OnTSMPPongReceived = func(pong packet.TSMPPongReply) {
- e.mu.Lock()
- defer e.mu.Unlock()
- cb := e.pongCallback[pong.Data]
- e.logf("wgengine: got TSMP pong %02x, peerAPIPort=%v; cb=%v", pong.Data, pong.PeerAPIPort, cb != nil)
- if cb != nil {
- go cb(pong)
- }
- }
- // wgdev takes ownership of tundev, will close it when closed.
- e.logf("Creating wireguard device...")
- e.wgdev = device.NewDevice(e.tundev, e.magicConn.Bind(), e.wgLogger.DeviceLogger)
- closePool.addFunc(e.wgdev.Close)
- closePool.addFunc(func() {
- if err := e.magicConn.Close(); err != nil {
- e.logf("error closing magicconn: %v", err)
- }
- })
- go func() {
- up := false
- for event := range e.tundev.EventsUpDown() {
- if event&tun.EventUp != 0 && !up {
- e.logf("external route: up")
- e.RequestStatus()
- up = true
- }
- if event&tun.EventDown != 0 && up {
- e.logf("external route: down")
- e.RequestStatus()
- up = false
- }
- }
- }()
- e.logf("Bringing wireguard device up...")
- if err := e.wgdev.Up(); err != nil {
- return nil, fmt.Errorf("wgdev.Up(): %w", err)
- }
- e.logf("Bringing router up...")
- if err := e.router.Up(); err != nil {
- return nil, err
- }
- // It's a little pointless to apply no-op settings here (they
- // should already be empty?), but it at least exercises the
- // router implementation early on.
- e.logf("Clearing router settings...")
- if err := e.router.Set(nil); err != nil {
- return nil, err
- }
- e.logf("Starting link monitor...")
- e.linkMon.Start()
- go e.pollResolver()
- e.logf("Engine created.")
- return e, nil
- }
- // echoRespondToAll is an inbound post-filter responding to all echo requests.
- func echoRespondToAll(p *packet.Parsed, t *tstun.Wrapper) filter.Response {
- if p.IsEchoRequest() {
- header := p.ICMP4Header()
- header.ToResponse()
- outp := packet.Generate(&header, p.Payload())
- t.InjectOutbound(outp)
- // We already responded to it, but it's not an error.
- // Proceed with regular delivery. (Since this code is only
- // used in fake mode, regular delivery just means throwing
- // it away. If this ever gets run in non-fake mode, you'll
- // get double responses to pings, which is an indicator you
- // shouldn't be doing that I guess.)
- return filter.Accept
- }
- return filter.Accept
- }
- // handleLocalPackets inspects packets coming from the local network
- // stack, and intercepts any packets that should be handled by
- // tailscaled directly. Other packets are allowed to proceed into the
- // main ACL filter.
- func (e *userspaceEngine) handleLocalPackets(p *packet.Parsed, t *tstun.Wrapper) filter.Response {
- if verdict := e.handleDNS(p, t); verdict == filter.Drop {
- // local DNS handled the packet.
- return filter.Drop
- }
- if runtime.GOOS == "darwin" || runtime.GOOS == "ios" {
- isLocalAddr, ok := e.isLocalAddr.Load().(func(netaddr.IP) bool)
- if !ok {
- e.logf("[unexpected] e.isLocalAddr was nil, can't check for loopback packet")
- } else if isLocalAddr(p.Dst.IP()) {
- // macOS NetworkExtension directs packets destined to the
- // tunnel's local IP address into the tunnel, instead of
- // looping back within the kernel network stack. We have to
- // notice that an outbound packet is actually destined for
- // ourselves, and loop it back into macOS.
- t.InjectInboundCopy(p.Buffer())
- return filter.Drop
- }
- }
- return filter.Accept
- }
- // handleDNS is an outbound pre-filter resolving Tailscale domains.
- func (e *userspaceEngine) handleDNS(p *packet.Parsed, t *tstun.Wrapper) filter.Response {
- if p.Dst.IP() == magicDNSIP && p.Dst.Port() == magicDNSPort && p.IPProto == ipproto.UDP {
- err := e.dns.EnqueueRequest(append([]byte(nil), p.Payload()...), p.Src)
- if err != nil {
- e.logf("dns: enqueue: %v", err)
- }
- return filter.Drop
- }
- return filter.Accept
- }
- // pollResolver reads responses from the DNS resolver and injects them inbound.
- func (e *userspaceEngine) pollResolver() {
- for {
- bs, to, err := e.dns.NextResponse()
- if err == resolver.ErrClosed {
- return
- }
- if err != nil {
- e.logf("dns: error: %v", err)
- continue
- }
- h := packet.UDP4Header{
- IP4Header: packet.IP4Header{
- Src: magicDNSIP,
- Dst: to.IP(),
- },
- SrcPort: magicDNSPort,
- DstPort: to.Port(),
- }
- hlen := h.Len()
- // TODO(dmytro): avoid this allocation without importing tstun quirks into dns.
- const offset = tstun.PacketStartOffset
- buf := make([]byte, offset+hlen+len(bs))
- copy(buf[offset+hlen:], bs)
- h.Marshal(buf[offset:])
- e.tundev.InjectInboundDirect(buf, offset)
- }
- }
- var (
- debugTrimWireguardEnv = os.Getenv("TS_DEBUG_TRIM_WIREGUARD")
- debugTrimWireguard, _ = strconv.ParseBool(debugTrimWireguardEnv)
- )
- // forceFullWireguardConfig reports whether we should give wireguard
- // our full network map, even for inactive peers
- //
- // TODO(bradfitz): remove this after our 1.0 launch; we don't want to
- // enable wireguard config trimming quite yet because it just landed
- // and we haven't got enough time testing it.
- func forceFullWireguardConfig(numPeers int) bool {
- // Did the user explicitly enable trimmming via the environment variable knob?
- if debugTrimWireguardEnv != "" {
- return !debugTrimWireguard
- }
- if opt := controlclient.TrimWGConfig(); opt != "" {
- return !opt.EqualBool(true)
- }
- // On iOS with large networks, it's critical, so turn on trimming.
- // Otherwise we run out of memory from wireguard-go goroutine stacks+buffers.
- // This will be the default later for all platforms and network sizes.
- if numPeers > 50 && version.OS() == "iOS" {
- return false
- }
- return false
- }
- // isTrimmablePeer reports whether p is a peer that we can trim out of the
- // network map.
- //
- // For implementation simplificy, we can only trim peers that have
- // only non-subnet AllowedIPs (an IPv4 /32 or IPv6 /128), which is the
- // common case for most peers. Subnet router nodes will just always be
- // created in the wireguard-go config.
- func isTrimmablePeer(p *wgcfg.Peer, numPeers int) bool {
- if forceFullWireguardConfig(numPeers) {
- return false
- }
- // AllowedIPs must all be single IPs, not subnets.
- for _, aip := range p.AllowedIPs {
- if !aip.IsSingleIP() {
- return false
- }
- }
- return true
- }
- // noteRecvActivity is called by magicsock when a packet has been
- // received for the peer with node key nk. Magicsock calls this no
- // more than every 10 seconds for a given peer.
- func (e *userspaceEngine) noteRecvActivity(nk key.NodePublic) {
- e.wgLock.Lock()
- defer e.wgLock.Unlock()
- if _, ok := e.recvActivityAt[nk]; !ok {
- // Not a trimmable peer we care about tracking. (See isTrimmablePeer)
- if e.trimmedNodes[nk] {
- e.logf("wgengine: [unexpected] noteReceiveActivity called on idle node %v that's not in recvActivityAt", nk.ShortString())
- }
- return
- }
- now := e.timeNow()
- e.recvActivityAt[nk] = now
- // As long as there's activity, periodically poll the engine to get
- // stats for the far away side effect of
- // ipn/ipnlocal.LocalBackend.parseWgStatusLocked to log activity, for
- // use in various admin dashboards.
- // This particularly matters on platforms without a connected GUI, as
- // the GUIs generally poll this enough to cause that logging. But
- // tailscaled alone did not, hence this.
- if e.lastStatusPollTime.IsZero() || now.Sub(e.lastStatusPollTime) >= statusPollInterval {
- e.lastStatusPollTime = now
- go e.RequestStatus()
- }
- // If the last activity time jumped a bunch (say, at least
- // half the idle timeout) then see if we need to reprogram
- // Wireguard. This could probably be just
- // lazyPeerIdleThreshold without the divide by 2, but
- // maybeReconfigWireguardLocked is cheap enough to call every
- // couple minutes (just not on every packet).
- if e.trimmedNodes[nk] {
- e.logf("wgengine: idle peer %v now active, reconfiguring wireguard", nk.ShortString())
- e.maybeReconfigWireguardLocked(nil)
- }
- }
- // isActiveSinceLocked reports whether the peer identified by (nk, ip)
- // has had a packet sent to or received from it since t.
- //
- // e.wgLock must be held.
- func (e *userspaceEngine) isActiveSinceLocked(nk key.NodePublic, ip netaddr.IP, t mono.Time) bool {
- if e.recvActivityAt[nk].After(t) {
- return true
- }
- timePtr, ok := e.sentActivityAt[ip]
- if !ok {
- return false
- }
- return timePtr.LoadAtomic().After(t)
- }
- // discoChanged are the set of peers whose disco keys have changed, implying they've restarted.
- // If a peer is in this set and was previously in the live wireguard config,
- // it needs to be first removed and then re-added to flush out its wireguard session key.
- // If discoChanged is nil or empty, this extra removal step isn't done.
- //
- // e.wgLock must be held.
- func (e *userspaceEngine) maybeReconfigWireguardLocked(discoChanged map[key.NodePublic]bool) error {
- if hook := e.testMaybeReconfigHook; hook != nil {
- hook()
- return nil
- }
- full := e.lastCfgFull
- e.wgLogger.SetPeers(full.Peers)
- // Compute a minimal config to pass to wireguard-go
- // based on the full config. Prune off all the peers
- // and only add the active ones back.
- min := full
- min.Peers = make([]wgcfg.Peer, 0, e.lastNMinPeers)
- // We'll only keep a peer around if it's been active in
- // the past 5 minutes. That's more than WireGuard's key
- // rotation time anyway so it's no harm if we remove it
- // later if it's been inactive.
- activeCutoff := e.timeNow().Add(-lazyPeerIdleThreshold)
- // Not all peers can be trimmed from the network map (see
- // isTrimmablePeer). For those are are trimmable, keep track of
- // their NodeKey and Tailscale IPs. These are the ones we'll need
- // to install tracking hooks for to watch their send/receive
- // activity.
- trackNodes := make([]key.NodePublic, 0, len(full.Peers))
- trackIPs := make([]netaddr.IP, 0, len(full.Peers))
- trimmedNodes := map[key.NodePublic]bool{} // TODO: don't re-alloc this map each time
- needRemoveStep := false
- for i := range full.Peers {
- p := &full.Peers[i]
- nk := p.PublicKey
- if !isTrimmablePeer(p, len(full.Peers)) {
- min.Peers = append(min.Peers, *p)
- if discoChanged[nk] {
- needRemoveStep = true
- }
- continue
- }
- trackNodes = append(trackNodes, nk)
- recentlyActive := false
- for _, cidr := range p.AllowedIPs {
- trackIPs = append(trackIPs, cidr.IP())
- recentlyActive = recentlyActive || e.isActiveSinceLocked(nk, cidr.IP(), activeCutoff)
- }
- if recentlyActive {
- min.Peers = append(min.Peers, *p)
- if discoChanged[nk] {
- needRemoveStep = true
- }
- } else {
- trimmedNodes[nk] = true
- }
- }
- e.lastNMinPeers = len(min.Peers)
- if !deephash.Update(&e.lastEngineSigTrim, &min, trimmedNodes, trackNodes, trackIPs) {
- // No changes
- return nil
- }
- e.trimmedNodes = trimmedNodes
- e.updateActivityMapsLocked(trackNodes, trackIPs)
- if needRemoveStep {
- minner := min
- minner.Peers = nil
- numRemove := 0
- for _, p := range min.Peers {
- if discoChanged[p.PublicKey] {
- numRemove++
- continue
- }
- minner.Peers = append(minner.Peers, p)
- }
- if numRemove > 0 {
- e.logf("wgengine: Reconfig: removing session keys for %d peers", numRemove)
- if err := wgcfg.ReconfigDevice(e.wgdev, &minner, e.logf); err != nil {
- e.logf("wgdev.Reconfig: %v", err)
- return err
- }
- }
- }
- e.logf("wgengine: Reconfig: configuring userspace wireguard config (with %d/%d peers)", len(min.Peers), len(full.Peers))
- if err := wgcfg.ReconfigDevice(e.wgdev, &min, e.logf); err != nil {
- e.logf("wgdev.Reconfig: %v", err)
- return err
- }
- return nil
- }
- // updateActivityMapsLocked updates the data structures used for tracking the activity
- // of wireguard peers that we might add/remove dynamically from the real config
- // as given to wireguard-go.
- //
- // e.wgLock must be held.
- func (e *userspaceEngine) updateActivityMapsLocked(trackNodes []key.NodePublic, trackIPs []netaddr.IP) {
- // Generate the new map of which nodekeys we want to track
- // receive times for.
- mr := map[key.NodePublic]mono.Time{} // TODO: only recreate this if set of keys changed
- for _, nk := range trackNodes {
- // Preserve old times in the new map, but also
- // populate map entries for new trackNodes values with
- // time.Time{} zero values. (Only entries in this map
- // are tracked, so the Time zero values allow it to be
- // tracked later)
- mr[nk] = e.recvActivityAt[nk]
- }
- e.recvActivityAt = mr
- oldTime := e.sentActivityAt
- e.sentActivityAt = make(map[netaddr.IP]*mono.Time, len(oldTime))
- oldFunc := e.destIPActivityFuncs
- e.destIPActivityFuncs = make(map[netaddr.IP]func(), len(oldFunc))
- updateFn := func(timePtr *mono.Time) func() {
- return func() {
- now := e.timeNow()
- old := timePtr.LoadAtomic()
- // How long's it been since we last sent a packet?
- elapsed := now.Sub(old)
- if old == 0 {
- // For our first packet, old is 0, which has indeterminate meaning.
- // Set elapsed to a big number (four score and seven years).
- elapsed = 762642 * time.Hour
- }
- if elapsed >= packetSendTimeUpdateFrequency {
- timePtr.StoreAtomic(now)
- }
- // On a big jump, assume we might no longer be in the wireguard
- // config and go check.
- if elapsed >= packetSendRecheckWireguardThreshold {
- e.wgLock.Lock()
- defer e.wgLock.Unlock()
- e.maybeReconfigWireguardLocked(nil)
- }
- }
- }
- for _, ip := range trackIPs {
- timePtr := oldTime[ip]
- if timePtr == nil {
- timePtr = new(mono.Time)
- }
- e.sentActivityAt[ip] = timePtr
- fn := oldFunc[ip]
- if fn == nil {
- fn = updateFn(timePtr)
- }
- e.destIPActivityFuncs[ip] = fn
- }
- e.tundev.SetDestIPActivityFuncs(e.destIPActivityFuncs)
- }
- // hasOverlap checks if there is a IPPrefix which is common amongst the two
- // provided slices.
- func hasOverlap(aips, rips []netaddr.IPPrefix) bool {
- for _, aip := range aips {
- for _, rip := range rips {
- if aip == rip {
- return true
- }
- }
- }
- return false
- }
- func (e *userspaceEngine) Reconfig(cfg *wgcfg.Config, routerCfg *router.Config, dnsCfg *dns.Config, debug *tailcfg.Debug) error {
- if routerCfg == nil {
- panic("routerCfg must not be nil")
- }
- if dnsCfg == nil {
- panic("dnsCfg must not be nil")
- }
- e.isLocalAddr.Store(tsaddr.NewContainsIPFunc(routerCfg.LocalAddrs))
- e.wgLock.Lock()
- defer e.wgLock.Unlock()
- e.lastDNSConfig = dnsCfg
- peerSet := make(map[key.NodePublic]struct{}, len(cfg.Peers))
- e.mu.Lock()
- e.peerSequence = e.peerSequence[:0]
- for _, p := range cfg.Peers {
- e.peerSequence = append(e.peerSequence, p.PublicKey)
- peerSet[p.PublicKey] = struct{}{}
- }
- e.mu.Unlock()
- listenPort := e.confListenPort
- if debug != nil && debug.RandomizeClientPort {
- listenPort = 0
- }
- isSubnetRouter := false
- if e.birdClient != nil {
- isSubnetRouter = hasOverlap(e.netMap.SelfNode.PrimaryRoutes, e.netMap.Hostinfo.RoutableIPs)
- }
- isSubnetRouterChanged := isSubnetRouter != e.lastIsSubnetRouter
- engineChanged := deephash.Update(&e.lastEngineSigFull, cfg)
- routerChanged := deephash.Update(&e.lastRouterSig, routerCfg, dnsCfg)
- if !engineChanged && !routerChanged && listenPort == e.magicConn.LocalPort() && !isSubnetRouterChanged {
- return ErrNoChanges
- }
- // TODO(bradfitz,danderson): maybe delete this isDNSIPOverTailscale
- // field and delete the resolver.ForwardLinkSelector hook and
- // instead have ipnlocal populate a map of DNS IP => linkName and
- // put that in the *dns.Config instead, and plumb it down to the
- // dns.Manager. Maybe also with isLocalAddr above.
- e.isDNSIPOverTailscale.Store(tsaddr.NewContainsIPFunc(dnsIPsOverTailscale(dnsCfg, routerCfg)))
- // See if any peers have changed disco keys, which means they've restarted.
- // If so, we need to update the wireguard-go/device.Device in two phases:
- // once without the node which has restarted, to clear its wireguard session key,
- // and a second time with it.
- discoChanged := make(map[key.NodePublic]bool)
- {
- prevEP := make(map[key.NodePublic]key.DiscoPublic)
- for i := range e.lastCfgFull.Peers {
- if p := &e.lastCfgFull.Peers[i]; !p.DiscoKey.IsZero() {
- prevEP[p.PublicKey] = p.DiscoKey
- }
- }
- for i := range cfg.Peers {
- p := &cfg.Peers[i]
- if p.DiscoKey.IsZero() {
- continue
- }
- pub := p.PublicKey
- if old, ok := prevEP[pub]; ok && old != p.DiscoKey {
- discoChanged[pub] = true
- e.logf("wgengine: Reconfig: %s changed from %q to %q", pub.ShortString(), old, p.DiscoKey)
- }
- }
- }
- e.lastCfgFull = *cfg.Clone()
- // Tell magicsock about the new (or initial) private key
- // (which is needed by DERP) before wgdev gets it, as wgdev
- // will start trying to handshake, which we want to be able to
- // go over DERP.
- if err := e.magicConn.SetPrivateKey(cfg.PrivateKey); err != nil {
- e.logf("wgengine: Reconfig: SetPrivateKey: %v", err)
- }
- e.magicConn.UpdatePeers(peerSet)
- e.magicConn.SetPreferredPort(listenPort)
- if err := e.maybeReconfigWireguardLocked(discoChanged); err != nil {
- return err
- }
- if routerChanged {
- e.logf("wgengine: Reconfig: configuring router")
- err := e.router.Set(routerCfg)
- health.SetRouterHealth(err)
- if err != nil {
- return err
- }
- // Keep DNS configuration after router configuration, as some
- // DNS managers refuse to apply settings if the device has no
- // assigned address.
- e.logf("wgengine: Reconfig: configuring DNS")
- err = e.dns.Set(*dnsCfg)
- health.SetDNSHealth(err)
- if err != nil {
- return err
- }
- }
- if isSubnetRouterChanged && e.birdClient != nil {
- e.logf("wgengine: Reconfig: configuring BIRD")
- var err error
- if isSubnetRouter {
- err = e.birdClient.EnableProtocol("tailscale")
- } else {
- err = e.birdClient.DisableProtocol("tailscale")
- }
- if err != nil {
- // Log but don't fail here.
- e.logf("wgengine: error configuring BIRD: %v", err)
- } else {
- e.lastIsSubnetRouter = isSubnetRouter
- }
- }
- e.logf("[v1] wgengine: Reconfig done")
- return nil
- }
- func (e *userspaceEngine) GetFilter() *filter.Filter {
- return e.tundev.GetFilter()
- }
- func (e *userspaceEngine) SetFilter(filt *filter.Filter) {
- e.tundev.SetFilter(filt)
- }
- func (e *userspaceEngine) SetStatusCallback(cb StatusCallback) {
- e.mu.Lock()
- defer e.mu.Unlock()
- e.statusCallback = cb
- }
- func (e *userspaceEngine) getStatusCallback() StatusCallback {
- e.mu.Lock()
- defer e.mu.Unlock()
- return e.statusCallback
- }
- var singleNewline = []byte{'\n'}
- var ErrEngineClosing = errors.New("engine closing; no status")
- func (e *userspaceEngine) getStatus() (*Status, error) {
- // Grab derpConns before acquiring wgLock to not violate lock ordering;
- // the DERPs method acquires magicsock.Conn.mu.
- // (See comment in userspaceEngine's declaration.)
- derpConns := e.magicConn.DERPs()
- e.wgLock.Lock()
- defer e.wgLock.Unlock()
- e.mu.Lock()
- closing := e.closing
- e.mu.Unlock()
- if closing {
- return nil, ErrEngineClosing
- }
- if e.wgdev == nil {
- // RequestStatus was invoked before the wgengine has
- // finished initializing. This can happen when wgegine
- // provides a callback to magicsock for endpoint
- // updates that calls RequestStatus.
- return nil, nil
- }
- pr, pw := io.Pipe()
- defer pr.Close() // to unblock writes on error path returns
- errc := make(chan error, 1)
- go func() {
- defer pw.Close()
- // TODO(apenwarr): get rid of silly uapi stuff for in-process comms
- // FIXME: get notified of status changes instead of polling.
- err := e.wgdev.IpcGetOperation(pw)
- if err != nil {
- err = fmt.Errorf("IpcGetOperation: %w", err)
- }
- errc <- err
- }()
- pp := make(map[key.NodePublic]ipnstate.PeerStatusLite)
- var p ipnstate.PeerStatusLite
- var hst1, hst2, n int64
- br := e.statusBufioReader
- if br != nil {
- br.Reset(pr)
- } else {
- br = bufio.NewReaderSize(pr, 1<<10)
- e.statusBufioReader = br
- }
- for {
- line, err := br.ReadSlice('\n')
- if err == io.EOF {
- break
- }
- if err != nil {
- return nil, fmt.Errorf("reading from UAPI pipe: %w", err)
- }
- line = bytes.TrimSuffix(line, singleNewline)
- k := line
- var v mem.RO
- if i := bytes.IndexByte(line, '='); i != -1 {
- k = line[:i]
- v = mem.B(line[i+1:])
- }
- switch string(k) {
- case "public_key":
- pk, err := key.ParseNodePublicUntyped(v)
- if err != nil {
- return nil, fmt.Errorf("IpcGetOperation: invalid key in line %q", line)
- }
- if !p.NodeKey.IsZero() {
- pp[p.NodeKey] = p
- }
- p = ipnstate.PeerStatusLite{NodeKey: pk}
- case "rx_bytes":
- n, err = mem.ParseInt(v, 10, 64)
- p.RxBytes = n
- if err != nil {
- return nil, fmt.Errorf("IpcGetOperation: rx_bytes invalid: %#v", line)
- }
- case "tx_bytes":
- n, err = mem.ParseInt(v, 10, 64)
- p.TxBytes = n
- if err != nil {
- return nil, fmt.Errorf("IpcGetOperation: tx_bytes invalid: %#v", line)
- }
- case "last_handshake_time_sec":
- hst1, err = mem.ParseInt(v, 10, 64)
- if err != nil {
- return nil, fmt.Errorf("IpcGetOperation: hst1 invalid: %#v", line)
- }
- case "last_handshake_time_nsec":
- hst2, err = mem.ParseInt(v, 10, 64)
- if err != nil {
- return nil, fmt.Errorf("IpcGetOperation: hst2 invalid: %#v", line)
- }
- if hst1 != 0 || hst2 != 0 {
- p.LastHandshake = time.Unix(hst1, hst2)
- } // else leave at time.IsZero()
- }
- }
- if !p.NodeKey.IsZero() {
- pp[p.NodeKey] = p
- }
- if err := <-errc; err != nil {
- return nil, fmt.Errorf("IpcGetOperation: %v", err)
- }
- e.mu.Lock()
- defer e.mu.Unlock()
- // Do two passes, one to calculate size and the other to populate.
- // This code is sensitive to allocations.
- npeers := 0
- for _, pk := range e.peerSequence {
- if _, ok := pp[pk]; ok { // ignore idle ones not in wireguard-go's config
- npeers++
- }
- }
- peers := make([]ipnstate.PeerStatusLite, 0, npeers)
- for _, pk := range e.peerSequence {
- if p, ok := pp[pk]; ok { // ignore idle ones not in wireguard-go's config
- peers = append(peers, p)
- }
- }
- return &Status{
- LocalAddrs: append([]tailcfg.Endpoint(nil), e.endpoints...),
- Peers: peers,
- DERPs: derpConns,
- }, nil
- }
- func (e *userspaceEngine) RequestStatus() {
- // This is slightly tricky. e.getStatus() can theoretically get
- // blocked inside wireguard for a while, and RequestStatus() is
- // sometimes called from a goroutine, so we don't want a lot of
- // them hanging around. On the other hand, requesting multiple
- // status updates simultaneously is pointless anyway; they will
- // all say the same thing.
- // Enqueue at most one request. If one is in progress already, this
- // adds one more to the queue. If one has been requested but not
- // started, it is a no-op.
- select {
- case e.reqCh <- struct{}{}:
- default:
- }
- // Dequeue at most one request. Another thread may have already
- // dequeued the request we enqueued above, which is fine, since the
- // information is guaranteed to be at least as recent as the current
- // call to RequestStatus().
- select {
- case <-e.reqCh:
- s, err := e.getStatus()
- if s == nil && err == nil {
- e.logf("[unexpected] RequestStatus: both s and err are nil")
- return
- }
- if cb := e.getStatusCallback(); cb != nil {
- cb(s, err)
- }
- default:
- }
- }
- func (e *userspaceEngine) Close() {
- e.mu.Lock()
- if e.closing {
- e.mu.Unlock()
- return
- }
- e.closing = true
- e.mu.Unlock()
- r := bufio.NewReader(strings.NewReader(""))
- e.wgdev.IpcSetOperation(r)
- e.magicConn.Close()
- e.linkMonUnregister()
- if e.linkMonOwned {
- e.linkMon.Close()
- }
- e.dns.Down()
- e.router.Close()
- e.wgdev.Close()
- e.tundev.Close()
- if e.birdClient != nil {
- e.birdClient.DisableProtocol("tailscale")
- e.birdClient.Close()
- }
- close(e.waitCh)
- }
- func (e *userspaceEngine) Wait() {
- <-e.waitCh
- }
- func (e *userspaceEngine) GetLinkMonitor() *monitor.Mon {
- return e.linkMon
- }
- // LinkChange signals a network change event. It's currently
- // (2021-03-03) only called on Android. On other platforms, linkMon
- // generates link change events for us.
- func (e *userspaceEngine) LinkChange(_ bool) {
- e.linkMon.InjectEvent()
- }
- func (e *userspaceEngine) linkChange(changed bool, cur *interfaces.State) {
- up := cur.AnyInterfaceUp()
- if !up {
- e.logf("LinkChange: all links down; pausing: %v", cur)
- } else if changed {
- e.logf("LinkChange: major, rebinding. New state: %v", cur)
- } else {
- e.logf("[v1] LinkChange: minor")
- }
- health.SetAnyInterfaceUp(up)
- e.magicConn.SetNetworkUp(up)
- if !up || changed {
- if err := e.dns.FlushCaches(); err != nil {
- e.logf("wgengine: dns flush failed after major link change: %v", err)
- }
- }
- // Hacky workaround for Linux DNS issue 2458: on
- // suspend/resume or whenever NetworkManager is started, it
- // nukes all systemd-resolved configs. So reapply our DNS
- // config on major link change.
- if runtime.GOOS == "linux" && changed {
- e.wgLock.Lock()
- dnsCfg := e.lastDNSConfig
- e.wgLock.Unlock()
- if dnsCfg != nil {
- if err := e.dns.Set(*dnsCfg); err != nil {
- e.logf("wgengine: error setting DNS config after major link change: %v", err)
- } else {
- e.logf("wgengine: set DNS config again after major link change")
- }
- }
- }
- why := "link-change-minor"
- if changed {
- why = "link-change-major"
- e.magicConn.Rebind()
- }
- e.magicConn.ReSTUN(why)
- }
- func (e *userspaceEngine) AddNetworkMapCallback(cb NetworkMapCallback) func() {
- e.mu.Lock()
- defer e.mu.Unlock()
- if e.networkMapCallbacks == nil {
- e.networkMapCallbacks = make(map[*someHandle]NetworkMapCallback)
- }
- h := new(someHandle)
- e.networkMapCallbacks[h] = cb
- return func() {
- e.mu.Lock()
- defer e.mu.Unlock()
- delete(e.networkMapCallbacks, h)
- }
- }
- func (e *userspaceEngine) SetNetInfoCallback(cb NetInfoCallback) {
- e.magicConn.SetNetInfoCallback(cb)
- }
- func (e *userspaceEngine) SetDERPMap(dm *tailcfg.DERPMap) {
- e.magicConn.SetDERPMap(dm)
- }
- func (e *userspaceEngine) SetNetworkMap(nm *netmap.NetworkMap) {
- e.magicConn.SetNetworkMap(nm)
- e.mu.Lock()
- e.netMap = nm
- callbacks := make([]NetworkMapCallback, 0, 4)
- for _, fn := range e.networkMapCallbacks {
- callbacks = append(callbacks, fn)
- }
- e.mu.Unlock()
- for _, fn := range callbacks {
- fn(nm)
- }
- }
- func (e *userspaceEngine) DiscoPublicKey() key.DiscoPublic {
- return e.magicConn.DiscoPublicKey()
- }
- func (e *userspaceEngine) UpdateStatus(sb *ipnstate.StatusBuilder) {
- st, err := e.getStatus()
- if err != nil {
- e.logf("wgengine: getStatus: %v", err)
- return
- }
- for _, ps := range st.Peers {
- sb.AddPeer(ps.NodeKey, &ipnstate.PeerStatus{
- RxBytes: int64(ps.RxBytes),
- TxBytes: int64(ps.TxBytes),
- LastHandshake: ps.LastHandshake,
- InEngine: true,
- })
- }
- e.magicConn.UpdateStatus(sb)
- }
- func (e *userspaceEngine) Ping(ip netaddr.IP, useTSMP bool, cb func(*ipnstate.PingResult)) {
- res := &ipnstate.PingResult{IP: ip.String()}
- peer, self, err := e.peerForIP(ip)
- if err != nil {
- e.logf("ping(%v): %v", ip, err)
- res.Err = err.Error()
- cb(res)
- return
- }
- if peer == nil {
- e.logf("ping(%v): no matching peer", ip)
- res.Err = "no matching peer"
- cb(res)
- return
- }
- if self {
- res.Err = fmt.Sprintf("%v is local Tailscale IP", ip)
- res.IsLocalIP = true
- cb(res)
- return
- }
- pingType := "disco"
- if useTSMP {
- pingType = "TSMP"
- }
- e.logf("ping(%v): sending %v ping to %v %v ...", ip, pingType, peer.Key.ShortString(), peer.ComputedName)
- if useTSMP {
- e.sendTSMPPing(ip, peer, res, cb)
- } else {
- e.magicConn.Ping(peer, res, cb)
- }
- }
- func (e *userspaceEngine) mySelfIPMatchingFamily(dst netaddr.IP) (src netaddr.IP, err error) {
- e.mu.Lock()
- defer e.mu.Unlock()
- if e.netMap == nil {
- return netaddr.IP{}, errors.New("no netmap")
- }
- for _, a := range e.netMap.Addresses {
- if a.IsSingleIP() && a.IP().BitLen() == dst.BitLen() {
- return a.IP(), nil
- }
- }
- if len(e.netMap.Addresses) == 0 {
- return netaddr.IP{}, errors.New("no self address in netmap")
- }
- return netaddr.IP{}, errors.New("no self address in netmap matching address family")
- }
- func (e *userspaceEngine) sendTSMPPing(ip netaddr.IP, peer *tailcfg.Node, res *ipnstate.PingResult, cb func(*ipnstate.PingResult)) {
- srcIP, err := e.mySelfIPMatchingFamily(ip)
- if err != nil {
- res.Err = err.Error()
- cb(res)
- return
- }
- var iph packet.Header
- if srcIP.Is4() {
- iph = packet.IP4Header{
- IPProto: ipproto.TSMP,
- Src: srcIP,
- Dst: ip,
- }
- } else {
- iph = packet.IP6Header{
- IPProto: ipproto.TSMP,
- Src: srcIP,
- Dst: ip,
- }
- }
- var data [8]byte
- crand.Read(data[:])
- expireTimer := time.AfterFunc(10*time.Second, func() {
- e.setTSMPPongCallback(data, nil)
- })
- t0 := time.Now()
- e.setTSMPPongCallback(data, func(pong packet.TSMPPongReply) {
- expireTimer.Stop()
- d := time.Since(t0)
- res.LatencySeconds = d.Seconds()
- res.NodeIP = ip.String()
- res.NodeName = peer.ComputedName
- res.PeerAPIPort = pong.PeerAPIPort
- cb(res)
- })
- var tsmpPayload [9]byte
- tsmpPayload[0] = byte(packet.TSMPTypePing)
- copy(tsmpPayload[1:], data[:])
- tsmpPing := packet.Generate(iph, tsmpPayload[:])
- e.tundev.InjectOutbound(tsmpPing)
- }
- func (e *userspaceEngine) setTSMPPongCallback(data [8]byte, cb func(packet.TSMPPongReply)) {
- e.mu.Lock()
- defer e.mu.Unlock()
- if e.pongCallback == nil {
- e.pongCallback = map[[8]byte]func(packet.TSMPPongReply){}
- }
- if cb == nil {
- delete(e.pongCallback, data)
- } else {
- e.pongCallback[data] = cb
- }
- }
- func (e *userspaceEngine) RegisterIPPortIdentity(ipport netaddr.IPPort, tsIP netaddr.IP) {
- e.mu.Lock()
- defer e.mu.Unlock()
- if e.tsIPByIPPort == nil {
- e.tsIPByIPPort = make(map[netaddr.IPPort]netaddr.IP)
- }
- e.tsIPByIPPort[ipport] = tsIP
- }
- func (e *userspaceEngine) UnregisterIPPortIdentity(ipport netaddr.IPPort) {
- e.mu.Lock()
- defer e.mu.Unlock()
- if e.tsIPByIPPort == nil {
- return
- }
- delete(e.tsIPByIPPort, ipport)
- }
- var whoIsSleeps = [...]time.Duration{
- 0,
- 10 * time.Millisecond,
- 20 * time.Millisecond,
- 50 * time.Millisecond,
- 100 * time.Millisecond,
- }
- func (e *userspaceEngine) WhoIsIPPort(ipport netaddr.IPPort) (tsIP netaddr.IP, ok bool) {
- // We currently have a registration race,
- // https://github.com/tailscale/tailscale/issues/1616,
- // so loop a few times for now waiting for the registration
- // to appear.
- // TODO(bradfitz,namansood): remove this once #1616 is fixed.
- for _, d := range whoIsSleeps {
- time.Sleep(d)
- e.mu.Lock()
- tsIP, ok = e.tsIPByIPPort[ipport]
- e.mu.Unlock()
- if ok {
- return tsIP, true
- }
- }
- return tsIP, false
- }
- // peerForIP returns the Node in the wireguard config
- // that's responsible for handling the given IP address.
- //
- // If none is found in the wireguard config but one is found in
- // the netmap, it's described in an error.
- //
- // If none is found in either place, (nil, nil) is returned.
- //
- // peerForIP acquires both e.mu and e.wgLock, but neither at the same
- // time.
- func (e *userspaceEngine) peerForIP(ip netaddr.IP) (n *tailcfg.Node, isSelf bool, err error) {
- e.mu.Lock()
- nm := e.netMap
- e.mu.Unlock()
- if nm == nil {
- return nil, false, errors.New("no network map")
- }
- // Check for exact matches before looking for subnet matches.
- var bestInNMPrefix netaddr.IPPrefix
- var bestInNM *tailcfg.Node
- for _, p := range nm.Peers {
- for _, a := range p.Addresses {
- if a.IP() == ip && a.IsSingleIP() && tsaddr.IsTailscaleIP(ip) {
- return p, false, nil
- }
- }
- for _, cidr := range p.AllowedIPs {
- if !cidr.Contains(ip) {
- continue
- }
- if bestInNMPrefix.IsZero() || cidr.Bits() > bestInNMPrefix.Bits() {
- bestInNMPrefix = cidr
- bestInNM = p
- }
- }
- }
- for _, a := range nm.Addresses {
- if a.IP() == ip && a.IsSingleIP() && tsaddr.IsTailscaleIP(ip) {
- return nm.SelfNode, true, nil
- }
- }
- e.wgLock.Lock()
- defer e.wgLock.Unlock()
- // TODO(bradfitz): this is O(n peers). Add ART to netaddr?
- var best netaddr.IPPrefix
- var bestKey key.NodePublic
- for _, p := range e.lastCfgFull.Peers {
- for _, cidr := range p.AllowedIPs {
- if !cidr.Contains(ip) {
- continue
- }
- if best.IsZero() || cidr.Bits() > best.Bits() {
- best = cidr
- bestKey = p.PublicKey
- }
- }
- }
- // And another pass. Probably better than allocating a map per peerForIP
- // call. But TODO(bradfitz): add a lookup map to netmap.NetworkMap.
- if !bestKey.IsZero() {
- for _, p := range nm.Peers {
- if p.Key == bestKey {
- return p, false, nil
- }
- }
- }
- if bestInNM == nil {
- return nil, false, nil
- }
- if bestInNMPrefix.Bits() == 0 {
- return nil, false, errors.New("exit node found but not enabled")
- }
- return nil, false, fmt.Errorf("node %q found, but not using its %v route", bestInNM.ComputedNameWithHost, bestInNMPrefix)
- }
- type closeOnErrorPool []func()
- func (p *closeOnErrorPool) add(c io.Closer) { *p = append(*p, func() { c.Close() }) }
- func (p *closeOnErrorPool) addFunc(fn func()) { *p = append(*p, fn) }
- func (p closeOnErrorPool) closeAllIfError(errp *error) {
- if *errp != nil {
- for _, closeFn := range p {
- closeFn()
- }
- }
- }
- // ipInPrefixes reports whether ip is in any of pp.
- func ipInPrefixes(ip netaddr.IP, pp []netaddr.IPPrefix) bool {
- for _, p := range pp {
- if p.Contains(ip) {
- return true
- }
- }
- return false
- }
- // dnsIPsOverTailscale returns the IPPrefixes of DNS resolver IPs that are
- // routed over Tailscale. The returned value does not contain duplicates is
- // not necessarily sorted.
- func dnsIPsOverTailscale(dnsCfg *dns.Config, routerCfg *router.Config) (ret []netaddr.IPPrefix) {
- m := map[netaddr.IP]bool{}
- add := func(resolvers []dnstype.Resolver) {
- for _, r := range resolvers {
- ip, err := netaddr.ParseIP(r.Addr)
- if err != nil {
- if ipp, err := netaddr.ParseIPPort(r.Addr); err == nil {
- ip = ipp.IP()
- } else {
- continue
- }
- }
- if ipInPrefixes(ip, routerCfg.Routes) && !ipInPrefixes(ip, routerCfg.LocalRoutes) {
- m[ip] = true
- }
- }
- }
- add(dnsCfg.DefaultResolvers)
- for _, resolvers := range dnsCfg.Routes {
- add(resolvers)
- }
- ret = make([]netaddr.IPPrefix, 0, len(m))
- for ip := range m {
- ret = append(ret, netaddr.IPPrefixFrom(ip, ip.BitLen()))
- }
- return ret
- }
- // fwdDNSLinkSelector is userspaceEngine's resolver.ForwardLinkSelector, to pick
- // which network interface to send DNS queries out of.
- type fwdDNSLinkSelector struct {
- ue *userspaceEngine
- tunName string
- }
- func (ls fwdDNSLinkSelector) PickLink(ip netaddr.IP) (linkName string) {
- if ls.ue.isDNSIPOverTailscale.Load().(func(netaddr.IP) bool)(ip) {
- return ls.tunName
- }
- return ""
- }
|