| 
									
										
										
										
											2023-01-27 13:37:20 -08:00
										 |  |  | // Copyright (c) Tailscale Inc & AUTHORS | 
					
						
							|  |  |  | // SPDX-License-Identifier: BSD-3-Clause | 
					
						
							| 
									
										
										
										
											2020-02-12 12:53:55 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | // Package monitor provides facilities for monitoring network | 
					
						
							| 
									
										
										
										
											2020-07-06 16:36:57 -07:00
										 |  |  | // interface and route changes. It primarily exists to know when | 
					
						
							|  |  |  | // portable devices move between different networks. | 
					
						
							| 
									
										
										
										
											2023-04-18 14:26:58 -07:00
										 |  |  | package netmon | 
					
						
							| 
									
										
										
										
											2020-02-12 12:53:55 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							| 
									
										
										
										
											2021-03-05 10:19:27 -08:00
										 |  |  | 	"encoding/json" | 
					
						
							| 
									
										
										
										
											2021-03-02 21:45:40 -08:00
										 |  |  | 	"errors" | 
					
						
							| 
									
										
										
											
												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-25 21:14:09 -07:00
										 |  |  | 	"net/netip" | 
					
						
							| 
									
										
										
										
											2021-03-25 10:23:13 -07:00
										 |  |  | 	"runtime" | 
					
						
							| 
									
										
										
										
											2020-02-17 09:00:38 -08:00
										 |  |  | 	"sync" | 
					
						
							| 
									
										
										
										
											2020-02-12 12:53:55 -08:00
										 |  |  | 	"time" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-01 07:55:30 -08:00
										 |  |  | 	"tailscale.com/net/interfaces" | 
					
						
							| 
									
										
										
										
											2020-02-14 19:23:16 -08:00
										 |  |  | 	"tailscale.com/types/logger" | 
					
						
							| 
									
										
										
										
											2023-09-01 19:56:18 -07:00
										 |  |  | 	"tailscale.com/util/clientmetric" | 
					
						
							| 
									
										
										
										
											2022-11-28 10:34:35 -08:00
										 |  |  | 	"tailscale.com/util/set" | 
					
						
							| 
									
										
										
										
											2020-02-12 12:53:55 -08:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-25 10:23:13 -07:00
										 |  |  | // 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 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-17 09:00:38 -08:00
										 |  |  | // message represents a message returned from an osMon. | 
					
						
							| 
									
										
										
										
											2020-07-06 16:36:57 -07:00
										 |  |  | type message interface { | 
					
						
							|  |  |  | 	// Ignore is whether we should ignore this message. | 
					
						
							|  |  |  | 	ignore() bool | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2020-02-12 12:53:55 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-17 09:00:38 -08:00
										 |  |  | // osMon is the interface that each operating system-specific | 
					
						
							|  |  |  | // implementation of the link monitor must implement. | 
					
						
							|  |  |  | type osMon interface { | 
					
						
							| 
									
										
										
										
											2020-02-12 12:53:55 -08:00
										 |  |  | 	Close() error | 
					
						
							| 
									
										
										
										
											2020-02-17 09:00:38 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// Receive returns a new network interface change message. It | 
					
						
							|  |  |  | 	// should block until there's either something to return, or | 
					
						
							|  |  |  | 	// until the osMon is closed. After a Close, the returned | 
					
						
							|  |  |  | 	// error is ignored. | 
					
						
							|  |  |  | 	Receive() (message, error) | 
					
						
							| 
									
										
										
										
											2022-04-20 12:03:50 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// IsInterestingInterface reports whether the provided interface should | 
					
						
							|  |  |  | 	// be considered for network change events. | 
					
						
							|  |  |  | 	IsInterestingInterface(iface string) bool | 
					
						
							| 
									
										
										
										
											2020-02-12 12:53:55 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-18 14:26:58 -07:00
										 |  |  | // Monitor represents a monitoring instance. | 
					
						
							|  |  |  | type Monitor struct { | 
					
						
							| 
									
										
										
										
											2020-02-12 12:53:55 -08:00
										 |  |  | 	logf   logger.Logf | 
					
						
							| 
									
										
										
										
											2023-08-23 15:04:58 -07:00
										 |  |  | 	om     osMon         // nil means not supported on this platform | 
					
						
							|  |  |  | 	change chan bool     // send false to wake poller, true to also force ChangeDeltas be sent | 
					
						
							| 
									
										
										
										
											2021-03-25 10:23:13 -07:00
										 |  |  | 	stop   chan struct{} // closed on Stop | 
					
						
							| 
									
										
										
										
											2021-02-27 19:33:21 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-23 11:48:05 -07:00
										 |  |  | 	// Things that must be set early, before use, | 
					
						
							|  |  |  | 	// and not change at runtime. | 
					
						
							|  |  |  | 	tsIfName string // tailscale interface name, if known/set ("tailscale0", "utun3", ...) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-25 10:23:13 -07:00
										 |  |  | 	mu         sync.Mutex // guards all following fields | 
					
						
							| 
									
										
										
										
											2023-04-18 14:26:58 -07:00
										 |  |  | 	cbs        set.HandleSet[ChangeFunc] | 
					
						
							| 
									
										
										
										
											2022-11-28 10:34:35 -08:00
										 |  |  | 	ruleDelCB  set.HandleSet[RuleDeleteCallback] | 
					
						
							| 
									
										
										
										
											2021-03-25 10:23:13 -07:00
										 |  |  | 	ifState    *interfaces.State | 
					
						
							|  |  |  | 	gwValid    bool       // whether gw and gwSelfIP are valid | 
					
						
							| 
									
										
										
											
												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-25 21:14:09 -07:00
										 |  |  | 	gw         netip.Addr // our gateway's IP | 
					
						
							|  |  |  | 	gwSelfIP   netip.Addr // our own IP address (that corresponds to gw) | 
					
						
							| 
									
										
										
										
											2020-02-17 09:00:38 -08:00
										 |  |  | 	started    bool | 
					
						
							| 
									
										
										
										
											2021-03-25 10:23:13 -07:00
										 |  |  | 	closed     bool | 
					
						
							| 
									
										
										
										
											2020-02-17 09:00:38 -08:00
										 |  |  | 	goroutines sync.WaitGroup | 
					
						
							| 
									
										
										
										
											2021-03-25 10:23:13 -07:00
										 |  |  | 	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 | 
					
						
							| 
									
										
										
										
											2020-02-12 12:53:55 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-18 14:26:58 -07:00
										 |  |  | // ChangeFunc is a callback function registered with Monitor that's called when the | 
					
						
							| 
									
										
										
										
											2023-08-23 10:05:21 -07:00
										 |  |  | // network changed. | 
					
						
							|  |  |  | type ChangeFunc func(*ChangeDelta) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // ChangeDelta describes the difference between two network states. | 
					
						
							|  |  |  | type ChangeDelta struct { | 
					
						
							| 
									
										
										
										
											2023-08-23 11:48:05 -07:00
										 |  |  | 	// Monitor is the network monitor that sent this delta. | 
					
						
							|  |  |  | 	Monitor *Monitor | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-23 10:05:21 -07:00
										 |  |  | 	// Old is the old interface state, if known. | 
					
						
							|  |  |  | 	// It's nil if the old state is unknown. | 
					
						
							|  |  |  | 	// Do not mutate it. | 
					
						
							|  |  |  | 	Old *interfaces.State | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// New is the new network state. | 
					
						
							|  |  |  | 	// It is always non-nil. | 
					
						
							|  |  |  | 	// Do not mutate it. | 
					
						
							|  |  |  | 	New *interfaces.State | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Major is our legacy boolean of whether the network changed in some major | 
					
						
							|  |  |  | 	// way. | 
					
						
							|  |  |  | 	// | 
					
						
							|  |  |  | 	// Deprecated: do not remove. As of 2023-08-23 we're in a renewed effort to | 
					
						
							|  |  |  | 	// remove it and ask specific qustions of ChangeDelta instead. Look at Old | 
					
						
							|  |  |  | 	// and New (or add methods to ChangeDelta) instead of using Major. | 
					
						
							|  |  |  | 	Major bool | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// TimeJumped is whether there was a big jump in wall time since the last | 
					
						
							|  |  |  | 	// time we checked. This is a hint that a mobile sleeping device might have | 
					
						
							|  |  |  | 	// come out of sleep. | 
					
						
							|  |  |  | 	TimeJumped bool | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// TODO(bradfitz): add some lazy cached fields here as needed with methods | 
					
						
							|  |  |  | 	// on *ChangeDelta to let callers ask specific questions | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2023-04-18 14:26:58 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-27 19:33:21 -08:00
										 |  |  | // New instantiates and starts a monitoring instance. | 
					
						
							| 
									
										
										
										
											2020-02-17 09:00:38 -08:00
										 |  |  | // The returned monitor is inactive until it's started by the Start method. | 
					
						
							| 
									
										
										
										
											2021-02-27 19:33:21 -08:00
										 |  |  | // Use RegisterChangeCallback to get notified of network changes. | 
					
						
							| 
									
										
										
										
											2023-04-18 14:26:58 -07:00
										 |  |  | func New(logf logger.Logf) (*Monitor, error) { | 
					
						
							| 
									
										
										
										
											2020-07-06 16:36:57 -07:00
										 |  |  | 	logf = logger.WithPrefix(logf, "monitor: ") | 
					
						
							| 
									
										
										
										
											2023-04-18 14:26:58 -07:00
										 |  |  | 	m := &Monitor{ | 
					
						
							| 
									
										
										
										
											2021-03-25 10:23:13 -07:00
										 |  |  | 		logf:     logf, | 
					
						
							| 
									
										
										
										
											2023-08-23 15:04:58 -07:00
										 |  |  | 		change:   make(chan bool, 1), | 
					
						
							| 
									
										
										
										
											2021-03-25 10:23:13 -07:00
										 |  |  | 		stop:     make(chan struct{}), | 
					
						
							|  |  |  | 		lastWall: wallTime(), | 
					
						
							| 
									
										
										
										
											2021-03-01 12:56:03 -08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	st, err := m.interfaceStateUncached() | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	m.ifState = st | 
					
						
							| 
									
										
										
										
											2021-03-02 21:45:40 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	m.om, err = newOSMon(logf, m) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if m.om == nil { | 
					
						
							|  |  |  | 		return nil, errors.New("newOSMon returned nil, nil") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-01 12:56:03 -08:00
										 |  |  | 	return m, nil | 
					
						
							| 
									
										
										
										
											2020-02-17 09:00:38 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-01 09:51:12 -08:00
										 |  |  | // InterfaceState returns the latest snapshot of the machine's network | 
					
						
							|  |  |  | // interfaces. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // The returned value is owned by Mon; it must not be modified. | 
					
						
							| 
									
										
										
										
											2023-04-18 14:26:58 -07:00
										 |  |  | func (m *Monitor) InterfaceState() *interfaces.State { | 
					
						
							| 
									
										
										
										
											2021-03-01 12:56:03 -08:00
										 |  |  | 	m.mu.Lock() | 
					
						
							|  |  |  | 	defer m.mu.Unlock() | 
					
						
							|  |  |  | 	return m.ifState | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-18 14:26:58 -07:00
										 |  |  | func (m *Monitor) interfaceStateUncached() (*interfaces.State, error) { | 
					
						
							| 
									
										
										
										
											2021-03-26 09:09:12 -07:00
										 |  |  | 	return interfaces.GetState() | 
					
						
							| 
									
										
										
										
											2021-03-01 07:55:30 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-23 11:48:05 -07:00
										 |  |  | // SetTailscaleInterfaceName sets the name of the Tailscale interface. For | 
					
						
							|  |  |  | // example, "tailscale0", "tun0", "utun3", etc. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // This must be called only early in tailscaled startup before the monitor is | 
					
						
							|  |  |  | // used. | 
					
						
							|  |  |  | func (m *Monitor) SetTailscaleInterfaceName(ifName string) { | 
					
						
							|  |  |  | 	m.tsIfName = ifName | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-15 13:58:10 -07:00
										 |  |  | // GatewayAndSelfIP returns the current network's default gateway, and | 
					
						
							|  |  |  | // the machine's default IP for that gateway. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // It's the same as interfaces.LikelyHomeRouterIP, but it caches the | 
					
						
							|  |  |  | // result until the monitor detects a network change. | 
					
						
							| 
									
										
										
										
											2023-04-18 14:26:58 -07:00
										 |  |  | func (m *Monitor) GatewayAndSelfIP() (gw, myIP netip.Addr, ok bool) { | 
					
						
							| 
									
										
										
										
											2021-03-15 13:58:10 -07:00
										 |  |  | 	m.mu.Lock() | 
					
						
							|  |  |  | 	defer m.mu.Unlock() | 
					
						
							|  |  |  | 	if m.gwValid { | 
					
						
							|  |  |  | 		return m.gw, m.gwSelfIP, true | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	gw, myIP, ok = interfaces.LikelyHomeRouterIP() | 
					
						
							| 
									
										
										
										
											2023-09-12 17:04:09 -04:00
										 |  |  | 	changed := false | 
					
						
							| 
									
										
										
										
											2021-03-15 13:58:10 -07:00
										 |  |  | 	if ok { | 
					
						
							| 
									
										
										
										
											2023-09-12 17:04:09 -04:00
										 |  |  | 		changed = m.gw != gw || m.gwSelfIP != myIP | 
					
						
							|  |  |  | 		m.gw, m.gwSelfIP = gw, myIP | 
					
						
							|  |  |  | 		m.gwValid = true | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if changed { | 
					
						
							|  |  |  | 		m.logf("gateway and self IP changed: gw=%v self=%v", m.gw, m.gwSelfIP) | 
					
						
							| 
									
										
										
										
											2021-03-15 13:58:10 -07:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	return gw, myIP, ok | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-27 19:33:21 -08:00
										 |  |  | // RegisterChangeCallback adds callback to the set of parties to be | 
					
						
							|  |  |  | // notified (in their own goroutine) when the network state changes. | 
					
						
							|  |  |  | // To remove this callback, call unregister (or close the monitor). | 
					
						
							| 
									
										
										
										
											2023-04-18 14:26:58 -07:00
										 |  |  | func (m *Monitor) RegisterChangeCallback(callback ChangeFunc) (unregister func()) { | 
					
						
							| 
									
										
										
										
											2021-02-27 19:33:21 -08:00
										 |  |  | 	m.mu.Lock() | 
					
						
							|  |  |  | 	defer m.mu.Unlock() | 
					
						
							| 
									
										
										
										
											2022-11-28 10:34:35 -08:00
										 |  |  | 	handle := m.cbs.Add(callback) | 
					
						
							| 
									
										
										
										
											2021-02-27 19:33:21 -08:00
										 |  |  | 	return func() { | 
					
						
							|  |  |  | 		m.mu.Lock() | 
					
						
							|  |  |  | 		defer m.mu.Unlock() | 
					
						
							|  |  |  | 		delete(m.cbs, handle) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-20 14:10:52 -07:00
										 |  |  | // RuleDeleteCallback is a callback when a Linux IP policy routing | 
					
						
							|  |  |  | // rule is deleted. The table is the table number (52, 253, 354) and | 
					
						
							|  |  |  | // priority is the priority order number (for Tailscale rules | 
					
						
							|  |  |  | // currently: 5210, 5230, 5250, 5270) | 
					
						
							|  |  |  | type RuleDeleteCallback func(table uint8, priority uint32) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // RegisterRuleDeleteCallback adds callback to the set of parties to be | 
					
						
							|  |  |  | // notified (in their own goroutine) when a Linux ip rule is deleted. | 
					
						
							|  |  |  | // To remove this callback, call unregister (or close the monitor). | 
					
						
							| 
									
										
										
										
											2023-04-18 14:26:58 -07:00
										 |  |  | func (m *Monitor) RegisterRuleDeleteCallback(callback RuleDeleteCallback) (unregister func()) { | 
					
						
							| 
									
										
										
										
											2021-07-20 14:10:52 -07:00
										 |  |  | 	m.mu.Lock() | 
					
						
							|  |  |  | 	defer m.mu.Unlock() | 
					
						
							| 
									
										
										
										
											2022-11-28 10:34:35 -08:00
										 |  |  | 	handle := m.ruleDelCB.Add(callback) | 
					
						
							| 
									
										
										
										
											2021-07-20 14:10:52 -07:00
										 |  |  | 	return func() { | 
					
						
							|  |  |  | 		m.mu.Lock() | 
					
						
							|  |  |  | 		defer m.mu.Unlock() | 
					
						
							|  |  |  | 		delete(m.ruleDelCB, handle) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-17 09:00:38 -08:00
										 |  |  | // Start starts the monitor. | 
					
						
							|  |  |  | // A monitor can only be started & closed once. | 
					
						
							| 
									
										
										
										
											2023-04-18 14:26:58 -07:00
										 |  |  | func (m *Monitor) Start() { | 
					
						
							| 
									
										
										
										
											2021-03-25 10:23:13 -07:00
										 |  |  | 	m.mu.Lock() | 
					
						
							|  |  |  | 	defer m.mu.Unlock() | 
					
						
							|  |  |  | 	if m.started || m.closed { | 
					
						
							|  |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	m.started = true | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-17 21:36:35 -07:00
										 |  |  | 	if shouldMonitorTimeJump { | 
					
						
							| 
									
										
										
										
											2021-03-25 10:23:13 -07:00
										 |  |  | 		m.wallTimer = time.AfterFunc(pollWallTimeInterval, m.pollWallTime) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if m.om == nil { | 
					
						
							|  |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	m.goroutines.Add(2) | 
					
						
							|  |  |  | 	go m.pump() | 
					
						
							|  |  |  | 	go m.debounce() | 
					
						
							| 
									
										
										
										
											2020-02-12 12:53:55 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-17 09:00:38 -08:00
										 |  |  | // Close closes the monitor. | 
					
						
							| 
									
										
										
										
											2023-04-18 14:26:58 -07:00
										 |  |  | func (m *Monitor) Close() error { | 
					
						
							| 
									
										
										
										
											2021-03-25 10:23:13 -07:00
										 |  |  | 	m.mu.Lock() | 
					
						
							|  |  |  | 	if m.closed { | 
					
						
							|  |  |  | 		m.mu.Unlock() | 
					
						
							|  |  |  | 		return nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	m.closed = true | 
					
						
							| 
									
										
										
										
											2020-02-12 12:53:55 -08:00
										 |  |  | 	close(m.stop) | 
					
						
							| 
									
										
										
										
											2021-03-25 10:23:13 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if m.wallTimer != nil { | 
					
						
							|  |  |  | 		m.wallTimer.Stop() | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-17 09:00:38 -08:00
										 |  |  | 	var err error | 
					
						
							|  |  |  | 	if m.om != nil { | 
					
						
							|  |  |  | 		err = m.om.Close() | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2021-03-25 10:23:13 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	started := m.started | 
					
						
							|  |  |  | 	m.mu.Unlock() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if started { | 
					
						
							| 
									
										
										
										
											2020-02-17 09:00:38 -08:00
										 |  |  | 		m.goroutines.Wait() | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return err | 
					
						
							| 
									
										
										
										
											2020-02-12 12:53:55 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-03 20:58:09 -08:00
										 |  |  | // InjectEvent forces the monitor to pretend there was a network | 
					
						
							|  |  |  | // change and re-check the state of the network. Any registered | 
					
						
							|  |  |  | // ChangeFunc callbacks will be called within the event coalescing | 
					
						
							|  |  |  | // period (under a fraction of a second). | 
					
						
							| 
									
										
										
										
											2023-04-18 14:26:58 -07:00
										 |  |  | func (m *Monitor) InjectEvent() { | 
					
						
							| 
									
										
										
										
											2021-03-03 20:58:09 -08:00
										 |  |  | 	select { | 
					
						
							| 
									
										
										
										
											2023-08-23 15:04:58 -07:00
										 |  |  | 	case m.change <- true: | 
					
						
							| 
									
										
										
										
											2021-03-03 20:58:09 -08:00
										 |  |  | 	default: | 
					
						
							|  |  |  | 		// Another change signal is already | 
					
						
							|  |  |  | 		// buffered. Debounce will wake up soon | 
					
						
							|  |  |  | 		// enough. | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-23 15:04:58 -07:00
										 |  |  | // Poll forces the monitor to pretend there was a network | 
					
						
							|  |  |  | // change and re-check the state of the network. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // This is like InjectEvent but only fires ChangeFunc callbacks | 
					
						
							|  |  |  | // if the network state differed at all. | 
					
						
							|  |  |  | func (m *Monitor) Poll() { | 
					
						
							|  |  |  | 	select { | 
					
						
							|  |  |  | 	case m.change <- false: | 
					
						
							|  |  |  | 	default: | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-18 14:26:58 -07:00
										 |  |  | func (m *Monitor) stopped() bool { | 
					
						
							| 
									
										
										
										
											2021-03-02 10:29:49 -08:00
										 |  |  | 	select { | 
					
						
							|  |  |  | 	case <-m.stop: | 
					
						
							|  |  |  | 		return true | 
					
						
							|  |  |  | 	default: | 
					
						
							|  |  |  | 		return false | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-12 12:53:55 -08:00
										 |  |  | // pump continuously retrieves messages from the connection, notifying | 
					
						
							|  |  |  | // the change channel of changes, and stopping when a stop is issued. | 
					
						
							| 
									
										
										
										
											2023-04-18 14:26:58 -07:00
										 |  |  | func (m *Monitor) pump() { | 
					
						
							| 
									
										
										
										
											2020-02-17 09:00:38 -08:00
										 |  |  | 	defer m.goroutines.Done() | 
					
						
							| 
									
										
										
										
											2021-03-02 10:29:49 -08:00
										 |  |  | 	for !m.stopped() { | 
					
						
							| 
									
										
										
										
											2020-07-06 16:36:57 -07:00
										 |  |  | 		msg, err := m.om.Receive() | 
					
						
							| 
									
										
										
										
											2020-02-12 12:53:55 -08:00
										 |  |  | 		if err != nil { | 
					
						
							| 
									
										
										
										
											2021-03-02 10:29:49 -08:00
										 |  |  | 			if m.stopped() { | 
					
						
							| 
									
										
										
										
											2020-02-12 12:53:55 -08:00
										 |  |  | 				return | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			// Keep retrying while we're not closed. | 
					
						
							| 
									
										
										
										
											2020-08-24 21:23:52 -07:00
										 |  |  | 			m.logf("error from link monitor: %v", err) | 
					
						
							| 
									
										
										
										
											2020-02-12 12:53:55 -08:00
										 |  |  | 			time.Sleep(time.Second) | 
					
						
							|  |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2021-07-20 14:10:52 -07:00
										 |  |  | 		if rdm, ok := msg.(ipRuleDeletedMessage); ok { | 
					
						
							|  |  |  | 			m.notifyRuleDeleted(rdm) | 
					
						
							|  |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2020-07-06 16:36:57 -07:00
										 |  |  | 		if msg.ignore() { | 
					
						
							|  |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2023-08-23 15:04:58 -07:00
										 |  |  | 		m.Poll() | 
					
						
							| 
									
										
										
										
											2020-02-12 12:53:55 -08:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-18 14:26:58 -07:00
										 |  |  | func (m *Monitor) notifyRuleDeleted(rdm ipRuleDeletedMessage) { | 
					
						
							| 
									
										
										
										
											2021-07-20 14:10:52 -07:00
										 |  |  | 	m.mu.Lock() | 
					
						
							|  |  |  | 	defer m.mu.Unlock() | 
					
						
							|  |  |  | 	for _, cb := range m.ruleDelCB { | 
					
						
							|  |  |  | 		go cb(rdm.table, rdm.priority) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-20 12:03:50 -07:00
										 |  |  | // isInterestingInterface reports whether the provided interface should be | 
					
						
							|  |  |  | // considered when checking for network state changes. | 
					
						
							|  |  |  | // The ips parameter should be the IPs of the provided interface. | 
					
						
							| 
									
										
										
										
											2023-04-18 14:26:58 -07:00
										 |  |  | func (m *Monitor) isInterestingInterface(i interfaces.Interface, ips []netip.Prefix) bool { | 
					
						
							| 
									
										
										
										
											2023-08-23 11:48:05 -07:00
										 |  |  | 	if !m.om.IsInterestingInterface(i.Name) { | 
					
						
							|  |  |  | 		return false | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return true | 
					
						
							| 
									
										
										
										
											2022-04-20 12:03:50 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-12 12:53:55 -08:00
										 |  |  | // debounce calls the callback function with a delay between events | 
					
						
							|  |  |  | // and exits when a stop is issued. | 
					
						
							| 
									
										
										
										
											2023-04-18 14:26:58 -07:00
										 |  |  | func (m *Monitor) debounce() { | 
					
						
							| 
									
										
										
										
											2020-02-17 09:00:38 -08:00
										 |  |  | 	defer m.goroutines.Done() | 
					
						
							| 
									
										
										
										
											2020-02-12 12:53:55 -08:00
										 |  |  | 	for { | 
					
						
							| 
									
										
										
										
											2023-08-23 15:04:58 -07:00
										 |  |  | 		var forceCallbacks bool | 
					
						
							| 
									
										
										
										
											2020-02-12 12:53:55 -08:00
										 |  |  | 		select { | 
					
						
							|  |  |  | 		case <-m.stop: | 
					
						
							|  |  |  | 			return | 
					
						
							| 
									
										
										
										
											2023-08-23 15:04:58 -07:00
										 |  |  | 		case forceCallbacks = <-m.change: | 
					
						
							| 
									
										
										
										
											2020-02-12 12:53:55 -08:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-23 15:04:58 -07:00
										 |  |  | 		if newState, err := m.interfaceStateUncached(); err != nil { | 
					
						
							| 
									
										
										
										
											2021-03-01 12:56:03 -08:00
										 |  |  | 			m.logf("interfaces.State: %v", err) | 
					
						
							|  |  |  | 		} else { | 
					
						
							| 
									
										
										
										
											2023-08-23 15:04:58 -07:00
										 |  |  | 			m.handlePotentialChange(newState, forceCallbacks) | 
					
						
							| 
									
										
										
										
											2021-02-27 19:33:21 -08:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2020-02-12 12:53:55 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		select { | 
					
						
							|  |  |  | 		case <-m.stop: | 
					
						
							|  |  |  | 			return | 
					
						
							| 
									
										
										
										
											2021-03-02 10:29:49 -08:00
										 |  |  | 		case <-time.After(250 * time.Millisecond): | 
					
						
							| 
									
										
										
										
											2020-02-12 12:53:55 -08:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-03-05 10:19:27 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-01 19:56:18 -07:00
										 |  |  | var ( | 
					
						
							|  |  |  | 	metricChangeEq       = clientmetric.NewCounter("netmon_link_change_eq") | 
					
						
							|  |  |  | 	metricChange         = clientmetric.NewCounter("netmon_link_change") | 
					
						
							|  |  |  | 	metricChangeTimeJump = clientmetric.NewCounter("netmon_link_change_timejump") | 
					
						
							|  |  |  | 	metricChangeMajor    = clientmetric.NewCounter("netmon_link_change_major") | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-23 15:04:58 -07:00
										 |  |  | // handlePotentialChange considers whether newState is different enough to wake | 
					
						
							|  |  |  | // up callers and updates the monitor's state if so. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // If forceCallbacks is true, they're always notified. | 
					
						
							|  |  |  | func (m *Monitor) handlePotentialChange(newState *interfaces.State, forceCallbacks bool) { | 
					
						
							|  |  |  | 	m.mu.Lock() | 
					
						
							|  |  |  | 	defer m.mu.Unlock() | 
					
						
							|  |  |  | 	oldState := m.ifState | 
					
						
							|  |  |  | 	timeJumped := shouldMonitorTimeJump && m.checkWallTimeAdvanceLocked() | 
					
						
							| 
									
										
										
										
											2023-08-23 11:48:05 -07:00
										 |  |  | 	if !timeJumped && !forceCallbacks && oldState.Equal(newState) { | 
					
						
							| 
									
										
										
										
											2023-08-23 15:04:58 -07:00
										 |  |  | 		// Exactly equal. Nothing to do. | 
					
						
							| 
									
										
										
										
											2023-09-01 19:56:18 -07:00
										 |  |  | 		metricChangeEq.Add(1) | 
					
						
							| 
									
										
										
										
											2023-08-23 15:04:58 -07:00
										 |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	delta := &ChangeDelta{ | 
					
						
							| 
									
										
										
										
											2023-08-23 11:48:05 -07:00
										 |  |  | 		Monitor:    m, | 
					
						
							| 
									
										
										
										
											2023-08-23 15:04:58 -07:00
										 |  |  | 		Old:        oldState, | 
					
						
							|  |  |  | 		New:        newState, | 
					
						
							|  |  |  | 		TimeJumped: timeJumped, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-23 11:48:05 -07:00
										 |  |  | 	delta.Major = m.IsMajorChangeFrom(oldState, newState) | 
					
						
							| 
									
										
										
										
											2023-08-23 15:04:58 -07:00
										 |  |  | 	if delta.Major { | 
					
						
							|  |  |  | 		m.gwValid = false | 
					
						
							|  |  |  | 		m.ifState = newState | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if s1, s2 := oldState.String(), delta.New.String(); s1 == s2 { | 
					
						
							|  |  |  | 			m.logf("[unexpected] network state changed, but stringification didn't: %v", s1) | 
					
						
							|  |  |  | 			m.logf("[unexpected] old: %s", jsonSummary(oldState)) | 
					
						
							|  |  |  | 			m.logf("[unexpected] new: %s", jsonSummary(newState)) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// See if we have a queued or new time jump signal. | 
					
						
							|  |  |  | 	if timeJumped { | 
					
						
							|  |  |  | 		m.resetTimeJumpedLocked() | 
					
						
							|  |  |  | 		if !delta.Major { | 
					
						
							|  |  |  | 			// Only log if it wasn't an interesting change. | 
					
						
							|  |  |  | 			m.logf("time jumped (probably wake from sleep); synthesizing major change event") | 
					
						
							|  |  |  | 			delta.Major = true | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2023-09-01 19:56:18 -07:00
										 |  |  | 	metricChange.Add(1) | 
					
						
							|  |  |  | 	if delta.Major { | 
					
						
							|  |  |  | 		metricChangeMajor.Add(1) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if delta.TimeJumped { | 
					
						
							|  |  |  | 		metricChangeTimeJump.Add(1) | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2023-08-23 15:04:58 -07:00
										 |  |  | 	for _, cb := range m.cbs { | 
					
						
							|  |  |  | 		go cb(delta) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-23 11:48:05 -07:00
										 |  |  | // IsMajorChangeFrom reports whether the transition from s1 to s2 is | 
					
						
							|  |  |  | // a "major" change, where major roughly means it's worth tearing down | 
					
						
							|  |  |  | // a bunch of connections and rebinding. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // TODO(bradiftz): tigten this definition. | 
					
						
							|  |  |  | func (m *Monitor) IsMajorChangeFrom(s1, s2 *interfaces.State) bool { | 
					
						
							|  |  |  | 	if s1 == nil && s2 == nil { | 
					
						
							|  |  |  | 		return false | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if s1 == nil || s2 == nil { | 
					
						
							|  |  |  | 		return true | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if s1.HaveV6 != s2.HaveV6 || | 
					
						
							|  |  |  | 		s1.HaveV4 != s2.HaveV4 || | 
					
						
							|  |  |  | 		s1.IsExpensive != s2.IsExpensive || | 
					
						
							|  |  |  | 		s1.DefaultRouteInterface != s2.DefaultRouteInterface || | 
					
						
							|  |  |  | 		s1.HTTPProxy != s2.HTTPProxy || | 
					
						
							|  |  |  | 		s1.PAC != s2.PAC { | 
					
						
							|  |  |  | 		return true | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	for iname, i := range s1.Interface { | 
					
						
							|  |  |  | 		if iname == m.tsIfName { | 
					
						
							|  |  |  | 			// Ignore changes in the Tailscale interface itself. | 
					
						
							|  |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		ips := s1.InterfaceIPs[iname] | 
					
						
							|  |  |  | 		if !m.isInterestingInterface(i, ips) { | 
					
						
							|  |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		i2, ok := s2.Interface[iname] | 
					
						
							|  |  |  | 		if !ok { | 
					
						
							|  |  |  | 			return true | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		ips2, ok := s2.InterfaceIPs[iname] | 
					
						
							|  |  |  | 		if !ok { | 
					
						
							|  |  |  | 			return true | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if !i.Equal(i2) || !prefixesMajorEqual(ips, ips2) { | 
					
						
							|  |  |  | 			return true | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-01-12 16:03:04 -08:00
										 |  |  | 	// Iterate over s2 in case there is a field in s2 that doesn't exist in s1 | 
					
						
							|  |  |  | 	for iname, i := range s2.Interface { | 
					
						
							|  |  |  | 		if iname == m.tsIfName { | 
					
						
							|  |  |  | 			// Ignore changes in the Tailscale interface itself. | 
					
						
							|  |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		ips := s2.InterfaceIPs[iname] | 
					
						
							|  |  |  | 		if !m.isInterestingInterface(i, ips) { | 
					
						
							|  |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		i1, ok := s1.Interface[iname] | 
					
						
							|  |  |  | 		if !ok { | 
					
						
							|  |  |  | 			return true | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		ips1, ok := s1.InterfaceIPs[iname] | 
					
						
							|  |  |  | 		if !ok { | 
					
						
							|  |  |  | 			return true | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if !i.Equal(i1) || !prefixesMajorEqual(ips, ips1) { | 
					
						
							|  |  |  | 			return true | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2023-08-23 11:48:05 -07:00
										 |  |  | 	return false | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // prefixesMajorEqual reports whether a and b are equal after ignoring | 
					
						
							|  |  |  | // boring things like link-local, loopback, and multicast addresses. | 
					
						
							|  |  |  | func prefixesMajorEqual(a, b []netip.Prefix) bool { | 
					
						
							|  |  |  | 	// trim returns a subslice of p with link local unicast, | 
					
						
							|  |  |  | 	// loopback, and multicast prefixes removed from the front. | 
					
						
							|  |  |  | 	trim := func(p []netip.Prefix) []netip.Prefix { | 
					
						
							|  |  |  | 		for len(p) > 0 { | 
					
						
							|  |  |  | 			a := p[0].Addr() | 
					
						
							|  |  |  | 			if a.IsLinkLocalUnicast() || a.IsLoopback() || a.IsMulticast() { | 
					
						
							|  |  |  | 				p = p[1:] | 
					
						
							|  |  |  | 				continue | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			break | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		return p | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	for { | 
					
						
							|  |  |  | 		a = trim(a) | 
					
						
							|  |  |  | 		b = trim(b) | 
					
						
							|  |  |  | 		if len(a) == 0 || len(b) == 0 { | 
					
						
							|  |  |  | 			return len(a) == 0 && len(b) == 0 | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if a[0] != b[0] { | 
					
						
							|  |  |  | 			return false | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		a, b = a[1:], b[1:] | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-16 16:27:57 -07:00
										 |  |  | func jsonSummary(x any) any { | 
					
						
							| 
									
										
										
										
											2021-03-05 10:19:27 -08:00
										 |  |  | 	j, err := json.Marshal(x) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return j | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-03-25 10:23:13 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | 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) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-18 14:26:58 -07:00
										 |  |  | func (m *Monitor) pollWallTime() { | 
					
						
							| 
									
										
										
										
											2021-03-25 10:23:13 -07:00
										 |  |  | 	m.mu.Lock() | 
					
						
							|  |  |  | 	defer m.mu.Unlock() | 
					
						
							|  |  |  | 	if m.closed { | 
					
						
							|  |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2022-04-17 21:36:35 -07:00
										 |  |  | 	if m.checkWallTimeAdvanceLocked() { | 
					
						
							| 
									
										
										
										
											2021-03-25 10:23:13 -07:00
										 |  |  | 		m.InjectEvent() | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	m.wallTimer.Reset(pollWallTimeInterval) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-17 21:36:35 -07:00
										 |  |  | // shouldMonitorTimeJump is whether we keep a regular periodic timer running in | 
					
						
							|  |  |  | // the background watching for jumps in wall time. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // We don't do this on mobile platforms for battery reasons, and because these | 
					
						
							|  |  |  | // platforms don't really sleep in the same way. | 
					
						
							|  |  |  | const shouldMonitorTimeJump = runtime.GOOS != "android" && runtime.GOOS != "ios" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // checkWallTimeAdvanceLocked reports whether wall time jumped more than 150% of | 
					
						
							|  |  |  | // pollWallTimeInterval, indicating we probably just came out of sleep. Once a | 
					
						
							|  |  |  | // time jump is detected it must be reset by calling resetTimeJumpedLocked. | 
					
						
							| 
									
										
										
										
											2023-04-18 14:26:58 -07:00
										 |  |  | func (m *Monitor) checkWallTimeAdvanceLocked() bool { | 
					
						
							| 
									
										
										
										
											2022-04-17 21:36:35 -07:00
										 |  |  | 	if !shouldMonitorTimeJump { | 
					
						
							|  |  |  | 		panic("unreachable") // if callers are correct | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2021-03-25 10:23:13 -07:00
										 |  |  | 	now := wallTime() | 
					
						
							|  |  |  | 	if now.Sub(m.lastWall) > pollWallTimeInterval*3/2 { | 
					
						
							| 
									
										
										
										
											2022-04-17 21:36:35 -07:00
										 |  |  | 		m.timeJumped = true // it is reset by debounce. | 
					
						
							| 
									
										
										
										
											2021-03-25 10:23:13 -07:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	m.lastWall = now | 
					
						
							| 
									
										
										
										
											2022-04-17 21:36:35 -07:00
										 |  |  | 	return m.timeJumped | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // resetTimeJumpedLocked consumes the signal set by checkWallTimeAdvanceLocked. | 
					
						
							| 
									
										
										
										
											2023-04-18 14:26:58 -07:00
										 |  |  | func (m *Monitor) resetTimeJumpedLocked() { | 
					
						
							| 
									
										
										
										
											2022-04-17 21:36:35 -07:00
										 |  |  | 	m.timeJumped = false | 
					
						
							| 
									
										
										
										
											2021-03-25 10:23:13 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2021-07-20 14:10:52 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | type ipRuleDeletedMessage struct { | 
					
						
							|  |  |  | 	table    uint8 | 
					
						
							|  |  |  | 	priority uint32 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (ipRuleDeletedMessage) ignore() bool { return true } |