mirror of
https://github.com/tailscale/tailscale.git
synced 2025-04-06 08:15:49 +00:00
wgengine/monitor: on wall time jump, synthesize network change event
... to force rebinds of TCP connections Fixes #1555 Updates tailscale/felicity#4 Signed-off-by: Brad Fitzpatrick <bradfitz@tailscale.com>
This commit is contained in:
parent
07bf4eb685
commit
a4c679e646
@ -10,6 +10,7 @@ package monitor
|
|||||||
import (
|
import (
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"errors"
|
"errors"
|
||||||
|
"runtime"
|
||||||
"sync"
|
"sync"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
@ -18,6 +19,13 @@ import (
|
|||||||
"tailscale.com/types/logger"
|
"tailscale.com/types/logger"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
// pollWallTimeInterval is how often we check the time to check
|
||||||
|
// for big jumps in wall (non-monotonic) time as a backup mechanism
|
||||||
|
// to get notified of a sleeping device waking back up.
|
||||||
|
// Usually there are also minor network change events on wake that let
|
||||||
|
// us check the wall time sooner than this.
|
||||||
|
const pollWallTimeInterval = 15 * time.Second
|
||||||
|
|
||||||
// message represents a message returned from an osMon.
|
// message represents a message returned from an osMon.
|
||||||
type message interface {
|
type message interface {
|
||||||
// Ignore is whether we should ignore this message.
|
// Ignore is whether we should ignore this message.
|
||||||
@ -50,18 +58,20 @@ type Mon struct {
|
|||||||
logf logger.Logf
|
logf logger.Logf
|
||||||
om osMon // nil means not supported on this platform
|
om osMon // nil means not supported on this platform
|
||||||
change chan struct{}
|
change chan struct{}
|
||||||
stop chan struct{}
|
stop chan struct{} // closed on Stop
|
||||||
|
|
||||||
mu sync.Mutex // guards cbs
|
mu sync.Mutex // guards all following fields
|
||||||
cbs map[*callbackHandle]ChangeFunc
|
cbs map[*callbackHandle]ChangeFunc
|
||||||
ifState *interfaces.State
|
ifState *interfaces.State
|
||||||
gwValid bool // whether gw and gwSelfIP are valid (cached)x
|
gwValid bool // whether gw and gwSelfIP are valid
|
||||||
gw netaddr.IP
|
gw netaddr.IP // our gateway's IP
|
||||||
gwSelfIP netaddr.IP
|
gwSelfIP netaddr.IP // our own IP address (that corresponds to gw)
|
||||||
|
|
||||||
onceStart sync.Once
|
|
||||||
started bool
|
started bool
|
||||||
|
closed bool
|
||||||
goroutines sync.WaitGroup
|
goroutines sync.WaitGroup
|
||||||
|
wallTimer *time.Timer // nil until Started; re-armed AfterFunc per tick
|
||||||
|
lastWall time.Time
|
||||||
|
timeJumped bool // whether we need to send a changed=true after a big time jump
|
||||||
}
|
}
|
||||||
|
|
||||||
// New instantiates and starts a monitoring instance.
|
// New instantiates and starts a monitoring instance.
|
||||||
@ -74,6 +84,7 @@ func New(logf logger.Logf) (*Mon, error) {
|
|||||||
cbs: map[*callbackHandle]ChangeFunc{},
|
cbs: map[*callbackHandle]ChangeFunc{},
|
||||||
change: make(chan struct{}, 1),
|
change: make(chan struct{}, 1),
|
||||||
stop: make(chan struct{}),
|
stop: make(chan struct{}),
|
||||||
|
lastWall: wallTime(),
|
||||||
}
|
}
|
||||||
st, err := m.interfaceStateUncached()
|
st, err := m.interfaceStateUncached()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@ -140,28 +151,54 @@ func (m *Mon) RegisterChangeCallback(callback ChangeFunc) (unregister func()) {
|
|||||||
// Start starts the monitor.
|
// Start starts the monitor.
|
||||||
// A monitor can only be started & closed once.
|
// A monitor can only be started & closed once.
|
||||||
func (m *Mon) Start() {
|
func (m *Mon) Start() {
|
||||||
m.onceStart.Do(func() {
|
m.mu.Lock()
|
||||||
if m.om == nil {
|
defer m.mu.Unlock()
|
||||||
|
if m.started || m.closed {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
m.started = true
|
m.started = true
|
||||||
|
|
||||||
|
switch runtime.GOOS {
|
||||||
|
case "ios", "android":
|
||||||
|
// For battery reasons, and because these platforms
|
||||||
|
// don't really sleep in the same way, don't poll
|
||||||
|
// for the wall time to detect for wake-for-sleep
|
||||||
|
// walltime jumps.
|
||||||
|
default:
|
||||||
|
m.wallTimer = time.AfterFunc(pollWallTimeInterval, m.pollWallTime)
|
||||||
|
}
|
||||||
|
|
||||||
|
if m.om == nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
m.goroutines.Add(2)
|
m.goroutines.Add(2)
|
||||||
go m.pump()
|
go m.pump()
|
||||||
go m.debounce()
|
go m.debounce()
|
||||||
})
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Close closes the monitor.
|
// Close closes the monitor.
|
||||||
// It may only be called once.
|
|
||||||
func (m *Mon) Close() error {
|
func (m *Mon) Close() error {
|
||||||
|
m.mu.Lock()
|
||||||
|
if m.closed {
|
||||||
|
m.mu.Unlock()
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
m.closed = true
|
||||||
close(m.stop)
|
close(m.stop)
|
||||||
|
|
||||||
|
if m.wallTimer != nil {
|
||||||
|
m.wallTimer.Stop()
|
||||||
|
}
|
||||||
|
|
||||||
var err error
|
var err error
|
||||||
if m.om != nil {
|
if m.om != nil {
|
||||||
err = m.om.Close()
|
err = m.om.Close()
|
||||||
}
|
}
|
||||||
// If it was previously started, wait for those goroutines to finish.
|
|
||||||
m.onceStart.Do(func() {})
|
started := m.started
|
||||||
if m.started {
|
m.mu.Unlock()
|
||||||
|
|
||||||
|
if started {
|
||||||
m.goroutines.Wait()
|
m.goroutines.Wait()
|
||||||
}
|
}
|
||||||
return err
|
return err
|
||||||
@ -227,9 +264,17 @@ func (m *Mon) debounce() {
|
|||||||
m.logf("interfaces.State: %v", err)
|
m.logf("interfaces.State: %v", err)
|
||||||
} else {
|
} else {
|
||||||
m.mu.Lock()
|
m.mu.Lock()
|
||||||
|
|
||||||
|
// See if we have a queued or new time jump signal.
|
||||||
|
m.checkWallTimeAdvanceLocked()
|
||||||
|
timeJumped := m.timeJumped
|
||||||
|
if timeJumped {
|
||||||
|
m.logf("time jumped (probably wake from sleep); synthesizing major change event")
|
||||||
|
}
|
||||||
|
|
||||||
oldState := m.ifState
|
oldState := m.ifState
|
||||||
changed := !curState.EqualFiltered(oldState, interfaces.FilterInteresting)
|
ifChanged := !curState.EqualFiltered(oldState, interfaces.FilterInteresting)
|
||||||
if changed {
|
if ifChanged {
|
||||||
m.gwValid = false
|
m.gwValid = false
|
||||||
m.ifState = curState
|
m.ifState = curState
|
||||||
|
|
||||||
@ -238,6 +283,10 @@ func (m *Mon) debounce() {
|
|||||||
jsonSummary(oldState), jsonSummary(curState))
|
jsonSummary(oldState), jsonSummary(curState))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
changed := ifChanged || timeJumped
|
||||||
|
if changed {
|
||||||
|
m.timeJumped = false
|
||||||
|
}
|
||||||
for _, cb := range m.cbs {
|
for _, cb := range m.cbs {
|
||||||
go cb(changed, m.ifState)
|
go cb(changed, m.ifState)
|
||||||
}
|
}
|
||||||
@ -259,3 +308,33 @@ func jsonSummary(x interface{}) interface{} {
|
|||||||
}
|
}
|
||||||
return j
|
return j
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func wallTime() time.Time {
|
||||||
|
// From time package's docs: "The canonical way to strip a
|
||||||
|
// monotonic clock reading is to use t = t.Round(0)."
|
||||||
|
return time.Now().Round(0)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Mon) pollWallTime() {
|
||||||
|
m.mu.Lock()
|
||||||
|
defer m.mu.Unlock()
|
||||||
|
if m.closed {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
m.checkWallTimeAdvanceLocked()
|
||||||
|
if m.timeJumped {
|
||||||
|
m.InjectEvent()
|
||||||
|
}
|
||||||
|
m.wallTimer.Reset(pollWallTimeInterval)
|
||||||
|
}
|
||||||
|
|
||||||
|
// checkWallTimeAdvanceLocked updates m.timeJumped, if wall time jumped
|
||||||
|
// more than 150% of pollWallTimeInterval, indicating we probably just
|
||||||
|
// came out of sleep.
|
||||||
|
func (m *Mon) checkWallTimeAdvanceLocked() {
|
||||||
|
now := wallTime()
|
||||||
|
if now.Sub(m.lastWall) > pollWallTimeInterval*3/2 {
|
||||||
|
m.timeJumped = true
|
||||||
|
}
|
||||||
|
m.lastWall = now
|
||||||
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user