mirror of
https://github.com/tailscale/tailscale.git
synced 2024-11-29 13:05:46 +00:00
wgengine/magicsock: disable regular STUNs for all platforms by default
Reduces background CPU & network. Updates #1034 Signed-off-by: Brad Fitzpatrick <bradfitz@tailscale.com>
This commit is contained in:
parent
49d00b6a28
commit
9541886856
@ -66,6 +66,7 @@ tailscale.com/cmd/tailscale dependencies: (generated by github.com/tailscale/dep
|
|||||||
💣 tailscale.com/syncs from tailscale.com/net/interfaces+
|
💣 tailscale.com/syncs from tailscale.com/net/interfaces+
|
||||||
tailscale.com/tailcfg from tailscale.com/cmd/tailscale/cli+
|
tailscale.com/tailcfg from tailscale.com/cmd/tailscale/cli+
|
||||||
W tailscale.com/tsconst from tailscale.com/net/interfaces
|
W tailscale.com/tsconst from tailscale.com/net/interfaces
|
||||||
|
tailscale.com/tstime from tailscale.com/wgengine/magicsock
|
||||||
tailscale.com/types/empty from tailscale.com/control/controlclient+
|
tailscale.com/types/empty from tailscale.com/control/controlclient+
|
||||||
tailscale.com/types/key from tailscale.com/cmd/tailscale/cli+
|
tailscale.com/types/key from tailscale.com/cmd/tailscale/cli+
|
||||||
tailscale.com/types/logger from tailscale.com/cmd/tailscale/cli+
|
tailscale.com/types/logger from tailscale.com/cmd/tailscale/cli+
|
||||||
|
@ -103,6 +103,7 @@ tailscale.com/cmd/tailscaled dependencies: (generated by github.com/tailscale/de
|
|||||||
💣 tailscale.com/syncs from tailscale.com/net/interfaces+
|
💣 tailscale.com/syncs from tailscale.com/net/interfaces+
|
||||||
tailscale.com/tailcfg from tailscale.com/control/controlclient+
|
tailscale.com/tailcfg from tailscale.com/control/controlclient+
|
||||||
W tailscale.com/tsconst from tailscale.com/net/interfaces
|
W tailscale.com/tsconst from tailscale.com/net/interfaces
|
||||||
|
tailscale.com/tstime from tailscale.com/wgengine/magicsock
|
||||||
tailscale.com/types/empty from tailscale.com/control/controlclient+
|
tailscale.com/types/empty from tailscale.com/control/controlclient+
|
||||||
tailscale.com/types/flagtype from tailscale.com/cmd/tailscaled
|
tailscale.com/types/flagtype from tailscale.com/cmd/tailscaled
|
||||||
tailscale.com/types/key from tailscale.com/derp+
|
tailscale.com/types/key from tailscale.com/derp+
|
||||||
|
@ -45,10 +45,10 @@
|
|||||||
"tailscale.com/net/stun"
|
"tailscale.com/net/stun"
|
||||||
"tailscale.com/syncs"
|
"tailscale.com/syncs"
|
||||||
"tailscale.com/tailcfg"
|
"tailscale.com/tailcfg"
|
||||||
|
"tailscale.com/tstime"
|
||||||
"tailscale.com/types/key"
|
"tailscale.com/types/key"
|
||||||
"tailscale.com/types/logger"
|
"tailscale.com/types/logger"
|
||||||
"tailscale.com/types/nettype"
|
"tailscale.com/types/nettype"
|
||||||
"tailscale.com/types/opt"
|
|
||||||
"tailscale.com/types/wgkey"
|
"tailscale.com/types/wgkey"
|
||||||
"tailscale.com/version"
|
"tailscale.com/version"
|
||||||
)
|
)
|
||||||
@ -182,6 +182,10 @@ type Conn struct {
|
|||||||
// scheduled to fire within derpCleanStaleInterval.
|
// scheduled to fire within derpCleanStaleInterval.
|
||||||
derpCleanupTimerArmed bool
|
derpCleanupTimerArmed bool
|
||||||
|
|
||||||
|
// periodicReSTUNTimer, when non-nil, is an AfterFunc timer
|
||||||
|
// that will call Conn.doPeriodicSTUN.
|
||||||
|
periodicReSTUNTimer *time.Timer
|
||||||
|
|
||||||
// endpointsUpdateActive indicates that updateEndpoints is
|
// endpointsUpdateActive indicates that updateEndpoints is
|
||||||
// currently running. It's used to deduplicate concurrent endpoint
|
// currently running. It's used to deduplicate concurrent endpoint
|
||||||
// update requests.
|
// update requests.
|
||||||
@ -196,6 +200,14 @@ type Conn struct {
|
|||||||
// change notifications.
|
// change notifications.
|
||||||
lastEndpoints []string
|
lastEndpoints []string
|
||||||
|
|
||||||
|
// lastEndpointsTime is the last time the endpoints were updated,
|
||||||
|
// even if there was no change.
|
||||||
|
lastEndpointsTime time.Time
|
||||||
|
|
||||||
|
// onEndpointRefreshed are funcs to run (in their own goroutines)
|
||||||
|
// when endpoints are refreshed.
|
||||||
|
onEndpointRefreshed map[*discoEndpoint]func()
|
||||||
|
|
||||||
// peerSet is the set of peers that are currently configured in
|
// peerSet is the set of peers that are currently configured in
|
||||||
// WireGuard. These are not used to filter inbound or outbound
|
// WireGuard. These are not used to filter inbound or outbound
|
||||||
// traffic at all, but only to track what state can be cleaned up
|
// traffic at all, but only to track what state can be cleaned up
|
||||||
@ -491,12 +503,6 @@ func (c *Conn) Start() {
|
|||||||
c.mu.Unlock()
|
c.mu.Unlock()
|
||||||
|
|
||||||
c.ReSTUN("initial")
|
c.ReSTUN("initial")
|
||||||
|
|
||||||
// We assume that LinkChange notifications are plumbed through well
|
|
||||||
// on our mobile clients, so don't do the timer thing to save radio/battery/CPU/etc.
|
|
||||||
if !version.IsMobile() {
|
|
||||||
go c.periodicReSTUN()
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// ignoreSTUNPackets sets a STUN packet processing func that does nothing.
|
// ignoreSTUNPackets sets a STUN packet processing func that does nothing.
|
||||||
@ -504,6 +510,17 @@ func (c *Conn) ignoreSTUNPackets() {
|
|||||||
c.stunReceiveFunc.Store(func([]byte, netaddr.IPPort) {})
|
c.stunReceiveFunc.Store(func([]byte, netaddr.IPPort) {})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// doPeriodicSTUN is called (in a new goroutine) by
|
||||||
|
// periodicReSTUNTimer when periodic STUNs are active.
|
||||||
|
func (c *Conn) doPeriodicSTUN() { c.ReSTUN("periodic") }
|
||||||
|
|
||||||
|
func (c *Conn) stopPeriodicReSTUNTimerLocked() {
|
||||||
|
if t := c.periodicReSTUNTimer; t != nil {
|
||||||
|
t.Stop()
|
||||||
|
c.periodicReSTUNTimer = nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// c.mu must NOT be held.
|
// c.mu must NOT be held.
|
||||||
func (c *Conn) updateEndpoints(why string) {
|
func (c *Conn) updateEndpoints(why string) {
|
||||||
defer func() {
|
defer func() {
|
||||||
@ -511,13 +528,37 @@ func (c *Conn) updateEndpoints(why string) {
|
|||||||
defer c.mu.Unlock()
|
defer c.mu.Unlock()
|
||||||
why := c.wantEndpointsUpdate
|
why := c.wantEndpointsUpdate
|
||||||
c.wantEndpointsUpdate = ""
|
c.wantEndpointsUpdate = ""
|
||||||
if why != "" && !c.closed {
|
if !c.closed {
|
||||||
go c.updateEndpoints(why)
|
if why != "" {
|
||||||
} else {
|
go c.updateEndpoints(why)
|
||||||
c.endpointsUpdateActive = false
|
return
|
||||||
c.muCond.Broadcast()
|
}
|
||||||
|
if c.shouldDoPeriodicReSTUNLocked() {
|
||||||
|
// Pick a random duration between 20
|
||||||
|
// and 26 seconds (just under 30s, a
|
||||||
|
// common UDP NAT timeout on Linux,
|
||||||
|
// etc)
|
||||||
|
d := tstime.RandomDurationBetween(20*time.Second, 26*time.Second)
|
||||||
|
if t := c.periodicReSTUNTimer; t != nil {
|
||||||
|
if debugReSTUNStopOnIdle {
|
||||||
|
c.logf("resetting existing periodicSTUN to run in %v", d)
|
||||||
|
}
|
||||||
|
t.Reset(d)
|
||||||
|
} else {
|
||||||
|
if debugReSTUNStopOnIdle {
|
||||||
|
c.logf("scheduling periodicSTUN to run in %v", d)
|
||||||
|
}
|
||||||
|
c.periodicReSTUNTimer = time.AfterFunc(d, c.doPeriodicSTUN)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if debugReSTUNStopOnIdle {
|
||||||
|
c.logf("periodic STUN idle")
|
||||||
|
}
|
||||||
|
c.stopPeriodicReSTUNTimerLocked()
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
c.endpointsUpdateActive = false
|
||||||
|
c.muCond.Broadcast()
|
||||||
}()
|
}()
|
||||||
c.logf("[v1] magicsock: starting endpoint update (%s)", why)
|
c.logf("[v1] magicsock: starting endpoint update (%s)", why)
|
||||||
|
|
||||||
@ -562,6 +603,12 @@ func (c *Conn) setEndpoints(endpoints []string, reasons map[string]string) (chan
|
|||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
|
c.lastEndpointsTime = time.Now()
|
||||||
|
for de, fn := range c.onEndpointRefreshed {
|
||||||
|
go fn()
|
||||||
|
delete(c.onEndpointRefreshed, de)
|
||||||
|
}
|
||||||
|
|
||||||
if stringsEqual(endpoints, c.lastEndpoints) {
|
if stringsEqual(endpoints, c.lastEndpoints) {
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
@ -1896,8 +1943,26 @@ func (c *Conn) enqueueCallMeMaybe(derpAddr netaddr.IPPort, de *discoEndpoint) {
|
|||||||
c.mu.Lock()
|
c.mu.Lock()
|
||||||
defer c.mu.Unlock()
|
defer c.mu.Unlock()
|
||||||
|
|
||||||
// TODO(bradfitz): do a fast/lite re-STUN if our STUN info is too old
|
if !c.lastEndpointsTime.After(time.Now().Add(-endpointsFreshEnoughDuration)) {
|
||||||
// Currently there's no "queue" despite the method name. There will be.
|
c.logf("magicsock: want call-me-maybe but endpoints stale; restunning")
|
||||||
|
if c.onEndpointRefreshed == nil {
|
||||||
|
c.onEndpointRefreshed = map[*discoEndpoint]func(){}
|
||||||
|
}
|
||||||
|
c.onEndpointRefreshed[de] = func() {
|
||||||
|
c.logf("magicsock: STUN done; sending call-me-maybe to %v %v", de.discoShort, de.publicKey.ShortString())
|
||||||
|
c.enqueueCallMeMaybe(derpAddr, de)
|
||||||
|
}
|
||||||
|
// TODO(bradfitz): make a new 'reSTUNQuickly' method
|
||||||
|
// that passes down a do-a-lite-netcheck flag down to
|
||||||
|
// netcheck that does 1 (or 2 max) STUN queries
|
||||||
|
// (UDP-only, not HTTPs) to find our port mapping to
|
||||||
|
// our home DERP and maybe one other. For now we do a
|
||||||
|
// "full" ReSTUN which may or may not be a full one
|
||||||
|
// (depending on age) and may do HTTPS timing queries
|
||||||
|
// (if UDP is blocked). Good enough for now.
|
||||||
|
go c.ReSTUN("refresh-for-peering")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
eps := make([]netaddr.IPPort, 0, len(c.lastEndpoints))
|
eps := make([]netaddr.IPPort, 0, len(c.lastEndpoints))
|
||||||
for _, ep := range c.lastEndpoints {
|
for _, ep := range c.lastEndpoints {
|
||||||
@ -2027,6 +2092,8 @@ func (c *Conn) SetPrivateKey(privateKey wgkey.Private) error {
|
|||||||
} else if newKey.IsZero() {
|
} else if newKey.IsZero() {
|
||||||
c.logf("magicsock: SetPrivateKey called (zeroed)")
|
c.logf("magicsock: SetPrivateKey called (zeroed)")
|
||||||
c.closeAllDerpLocked("zero-private-key")
|
c.closeAllDerpLocked("zero-private-key")
|
||||||
|
c.stopPeriodicReSTUNTimerLocked()
|
||||||
|
c.onEndpointRefreshed = nil
|
||||||
} else {
|
} else {
|
||||||
c.logf("magicsock: SetPrivateKey called (changed)")
|
c.logf("magicsock: SetPrivateKey called (changed)")
|
||||||
c.closeAllDerpLocked("new-private-key")
|
c.closeAllDerpLocked("new-private-key")
|
||||||
@ -2299,6 +2366,7 @@ func (c *Conn) Close() error {
|
|||||||
if c.derpCleanupTimerArmed {
|
if c.derpCleanupTimerArmed {
|
||||||
c.derpCleanupTimer.Stop()
|
c.derpCleanupTimer.Stop()
|
||||||
}
|
}
|
||||||
|
c.stopPeriodicReSTUNTimerLocked()
|
||||||
|
|
||||||
for _, ep := range c.endpointOfDisco {
|
for _, ep := range c.endpointOfDisco {
|
||||||
ep.stopAndReset()
|
ep.stopAndReset()
|
||||||
@ -2347,73 +2415,36 @@ func (c *Conn) goroutinesRunningLocked() bool {
|
|||||||
|
|
||||||
func maxIdleBeforeSTUNShutdown() time.Duration {
|
func maxIdleBeforeSTUNShutdown() time.Duration {
|
||||||
if debugReSTUNStopOnIdle {
|
if debugReSTUNStopOnIdle {
|
||||||
return time.Minute
|
return 45 * time.Second
|
||||||
}
|
}
|
||||||
return 5 * time.Minute
|
return sessionActiveTimeout
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *Conn) shouldDoPeriodicReSTUN() bool {
|
func (c *Conn) shouldDoPeriodicReSTUNLocked() bool {
|
||||||
if c.networkDown() {
|
if c.networkDown() {
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
if len(c.peerSet) == 0 || c.privateKey.IsZero() {
|
||||||
c.mu.Lock()
|
// If no peers, not worth doing.
|
||||||
defer c.mu.Unlock()
|
// Also don't if there's no key (not running).
|
||||||
if len(c.peerSet) == 0 {
|
|
||||||
// No peers, so not worth doing.
|
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
// If it turns out this optimization was a mistake, we can
|
|
||||||
// override it from the control server without waiting for a
|
|
||||||
// new software rollout:
|
|
||||||
if c.netMap != nil && c.netMap.Debug != nil && c.netMap.Debug.ForceBackgroundSTUN && !debugReSTUNStopOnIdle {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
if f := c.idleFunc; f != nil {
|
if f := c.idleFunc; f != nil {
|
||||||
idleFor := f()
|
idleFor := f()
|
||||||
if debugReSTUNStopOnIdle {
|
if debugReSTUNStopOnIdle {
|
||||||
c.logf("magicsock: periodicReSTUN: idle for %v", idleFor.Round(time.Second))
|
c.logf("magicsock: periodicReSTUN: idle for %v", idleFor.Round(time.Second))
|
||||||
}
|
}
|
||||||
if idleFor > maxIdleBeforeSTUNShutdown() {
|
if idleFor > maxIdleBeforeSTUNShutdown() {
|
||||||
if debugReSTUNStopOnIdle || version.IsMobile() { // TODO: make this unconditional later
|
if c.netMap != nil && c.netMap.Debug != nil && c.netMap.Debug.ForceBackgroundSTUN {
|
||||||
return false
|
// Overridden by control.
|
||||||
|
return true
|
||||||
}
|
}
|
||||||
|
return false
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *Conn) periodicReSTUN() {
|
|
||||||
prand := rand.New(rand.NewSource(time.Now().UnixNano()))
|
|
||||||
dur := func() time.Duration {
|
|
||||||
// Just under 30s, a common UDP NAT timeout (Linux at least)
|
|
||||||
return time.Duration(20+prand.Intn(7)) * time.Second
|
|
||||||
}
|
|
||||||
timer := time.NewTimer(dur())
|
|
||||||
defer timer.Stop()
|
|
||||||
var lastIdleState opt.Bool
|
|
||||||
for {
|
|
||||||
select {
|
|
||||||
case <-c.donec:
|
|
||||||
return
|
|
||||||
case <-timer.C:
|
|
||||||
doReSTUN := c.shouldDoPeriodicReSTUN()
|
|
||||||
if !lastIdleState.EqualBool(doReSTUN) {
|
|
||||||
if doReSTUN {
|
|
||||||
c.logf("[v1] magicsock: periodicReSTUN enabled")
|
|
||||||
} else {
|
|
||||||
c.logf("[v1] magicsock: periodicReSTUN disabled due to inactivity")
|
|
||||||
}
|
|
||||||
lastIdleState.Set(doReSTUN)
|
|
||||||
}
|
|
||||||
if doReSTUN {
|
|
||||||
c.ReSTUN("periodic")
|
|
||||||
}
|
|
||||||
timer.Reset(dur())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// ReSTUN triggers an address discovery.
|
// ReSTUN triggers an address discovery.
|
||||||
// The provided why string is for debug logging only.
|
// The provided why string is for debug logging only.
|
||||||
func (c *Conn) ReSTUN(why string) {
|
func (c *Conn) ReSTUN(why string) {
|
||||||
@ -2880,6 +2911,8 @@ type pendingCLIPing struct {
|
|||||||
const (
|
const (
|
||||||
// sessionActiveTimeout is how long since the last activity we
|
// sessionActiveTimeout is how long since the last activity we
|
||||||
// try to keep an established discoEndpoint peering alive.
|
// try to keep an established discoEndpoint peering alive.
|
||||||
|
// It's also the idle time at which we stop doing STUN queries to
|
||||||
|
// keep NAT mappings alive.
|
||||||
sessionActiveTimeout = 2 * time.Minute
|
sessionActiveTimeout = 2 * time.Minute
|
||||||
|
|
||||||
// upgradeInterval is how often we try to upgrade to a better path
|
// upgradeInterval is how often we try to upgrade to a better path
|
||||||
@ -2915,6 +2948,11 @@ type pendingCLIPing struct {
|
|||||||
// derpCleanStaleInterval is how often cleanStaleDerp runs when there
|
// derpCleanStaleInterval is how often cleanStaleDerp runs when there
|
||||||
// are potentially-stale DERP connections to close.
|
// are potentially-stale DERP connections to close.
|
||||||
derpCleanStaleInterval = 15 * time.Second
|
derpCleanStaleInterval = 15 * time.Second
|
||||||
|
|
||||||
|
// endpointsFreshEnoughDuration is how long we consider a
|
||||||
|
// STUN-derived endpoint valid for. UDP NAT mappings typically
|
||||||
|
// expire at 30 seconds, so this is a few seconds shy of that.
|
||||||
|
endpointsFreshEnoughDuration = 27 * time.Second
|
||||||
)
|
)
|
||||||
|
|
||||||
// endpointState is some state and history for a specific endpoint of
|
// endpointState is some state and history for a specific endpoint of
|
||||||
|
Loading…
Reference in New Issue
Block a user