ipn: fix some mutex/ownership issues

Signed-off-by: Brad Fitzpatrick <bradfitz@tailscale.com>
This commit is contained in:
Brad Fitzpatrick 2020-02-25 12:30:28 -08:00 committed by Brad Fitzpatrick
parent 6d2ac01464
commit eefafad9f8
3 changed files with 76 additions and 53 deletions

View File

@ -29,6 +29,7 @@ func (s State) String() string {
"Stopped", "Starting", "Running"}[s] "Stopped", "Starting", "Running"}[s]
} }
// EngineStatus contains WireGuard engine stats.
type EngineStatus struct { type EngineStatus struct {
RBytes, WBytes wgengine.ByteCount RBytes, WBytes wgengine.ByteCount
NumLive int NumLive int
@ -99,18 +100,24 @@ type Options struct {
Notify func(Notify) `json:"-"` Notify func(Notify) `json:"-"`
} }
// Backend is the interface between Tailscale frontends
// (e.g. cmd/tailscale, iOS/MacOS/Windows GUIs) and the tailscale
// backend (e.g. cmd/tailscaled) running on the same machine.
// (It has nothing to do with the interface between the backends
// and the cloud control plane.)
type Backend interface { type Backend interface {
// Start or restart the backend, because a new Handle has connected. // Start starts or restarts the backend, typically when a
Start(opts Options) error // frontend client connects.
Start(Options) error
// StartLoginInteractive requests to start a new interactive login // StartLoginInteractive requests to start a new interactive login
// flow. This should trigger a new BrowseToURL notification // flow. This should trigger a new BrowseToURL notification
// eventually. // eventually.
StartLoginInteractive() StartLoginInteractive()
// Logout terminates the current login session and stop the // Logout terminates the current login session and stops the
// wireguard engine. // wireguard engine.
Logout() Logout()
// SetPrefs install a new set of user preferences, including // SetPrefs installs a new set of user preferences, including
// WantRunning. This may cause the wireguard engine to // WantRunning. This may cause the wireguard engine to
// reconfigure or stop. // reconfigure or stop.
SetPrefs(new *Prefs) SetPrefs(new *Prefs)
// RequestEngineStatus polls for an update from the wireguard // RequestEngineStatus polls for an update from the wireguard

View File

@ -15,7 +15,7 @@ import (
type Handle struct { type Handle struct {
frontendLogID string frontendLogID string
b Backend b Backend
xnotify func(n Notify) xnotify func(Notify)
logf logger.Logf logf logger.Logf
// Mutex protects everything below // Mutex protects everything below

View File

@ -28,11 +28,9 @@ import (
// from the cloud to the local WireGuard engine. // from the cloud to the local WireGuard engine.
type LocalBackend struct { type LocalBackend struct {
logf logger.Logf logf logger.Logf
notify func(Notify)
c *controlclient.Client // TODO: appears to be (inconsistently) guarded by mu
e wgengine.Engine e wgengine.Engine
store StateStore store StateStore
serverURL string serverURL string // tailcontrol URL
backendLogID string backendLogID string
portpoll *portlist.Poller // may be nil portpoll *portlist.Poller // may be nil
newDecompressor func() (controlclient.Decompressor, error) newDecompressor func() (controlclient.Decompressor, error)
@ -40,13 +38,15 @@ type LocalBackend struct {
// The mutex protects the following elements. // The mutex protects the following elements.
mu sync.Mutex mu sync.Mutex
notify func(Notify)
c *controlclient.Client // TODO: appears to be (inconsistently) guarded by mu
stateKey StateKey stateKey StateKey
prefs *Prefs prefs *Prefs
state State state State
hiCache *tailcfg.Hostinfo hiCache *tailcfg.Hostinfo
netMapCache *controlclient.NetworkMap netMapCache *controlclient.NetworkMap // TODO: many uses without holding mu
engineStatus EngineStatus engineStatus EngineStatus // TODO: many uses without holding mu
endPoints []string endPoints []string // TODO: many uses without holding mu
blocked bool blocked bool
authURL string authURL string
interact int interact int
@ -125,17 +125,6 @@ func (b *LocalBackend) Start(opts Options) error {
return errors.New("no state key or prefs provided") return errors.New("no state key or prefs provided")
} }
if b.c != nil {
// TODO(apenwarr): avoid the need to reinit controlclient.
// This will trigger a full relogin/reconfigure cycle every
// time a Handle reconnects to the backend. Ideally, we
// would send the new Prefs and everything would get back
// into sync with the minimal changes. But that's not how it
// is right now, which is a sign that the code is still too
// complicated.
b.c.Shutdown()
}
if opts.Prefs != nil { if opts.Prefs != nil {
b.logf("Start: %v\n", opts.Prefs.Pretty()) b.logf("Start: %v\n", opts.Prefs.Pretty())
} else { } else {
@ -147,13 +136,25 @@ func (b *LocalBackend) Start(opts Options) error {
hi.FrontendLogID = opts.FrontendLogID hi.FrontendLogID = opts.FrontendLogID
b.mu.Lock() b.mu.Lock()
if b.c != nil {
// TODO(apenwarr): avoid the need to reinit controlclient.
// This will trigger a full relogin/reconfigure cycle every
// time a Handle reconnects to the backend. Ideally, we
// would send the new Prefs and everything would get back
// into sync with the minimal changes. But that's not how it
// is right now, which is a sign that the code is still too
// complicated.
b.c.Shutdown()
}
if b.hiCache != nil { if b.hiCache != nil {
hi.Services = b.hiCache.Services // keep any previous session hi.Services = b.hiCache.Services // keep any previous session
} }
b.hiCache = hi b.hiCache = hi
b.state = NoState b.state = NoState
if err := b.loadStateWithLock(opts.StateKey, opts.Prefs, opts.LegacyConfigPath); err != nil { if err := b.loadStateLocked(opts.StateKey, opts.Prefs, opts.LegacyConfigPath); err != nil {
b.mu.Unlock() b.mu.Unlock()
return fmt.Errorf("loading requested state: %v", err) return fmt.Errorf("loading requested state: %v", err)
} }
@ -193,51 +194,51 @@ func (b *LocalBackend) Start(opts Options) error {
cli.UpdateEndpoints(0, b.endPoints) cli.UpdateEndpoints(0, b.endPoints)
} }
cli.SetStatusFunc(func(new controlclient.Status) { cli.SetStatusFunc(func(newSt controlclient.Status) {
if new.LoginFinished != nil { if newSt.LoginFinished != nil {
// Auth completed, unblock the engine // Auth completed, unblock the engine
b.blockEngineUpdates(false) b.blockEngineUpdates(false)
b.authReconfig() b.authReconfig()
b.send(Notify{LoginFinished: &empty.Message{}}) b.send(Notify{LoginFinished: &empty.Message{}})
} }
if new.Persist != nil { if newSt.Persist != nil {
persist := *new.Persist // copy persist := *newSt.Persist // copy
b.prefs.Persist = &persist b.prefs.Persist = &persist
if b.stateKey != "" { if b.stateKey != "" { // TODO: accessed without b.mu held?
if err := b.store.WriteState(b.stateKey, b.prefs.ToBytes()); err != nil { if err := b.store.WriteState(b.stateKey, b.prefs.ToBytes()); err != nil {
b.logf("Failed to save new controlclient state: %v", err) b.logf("Failed to save new controlclient state: %v", err)
} }
} }
b.send(Notify{Prefs: b.prefs.Copy()}) b.send(Notify{Prefs: b.prefs.Copy()})
} }
if new.NetMap != nil { if newSt.NetMap != nil {
if b.netMapCache != nil && b.cmpDiff != nil { if b.netMapCache != nil && b.cmpDiff != nil {
s1 := strings.Split(b.netMapCache.Concise(), "\n") s1 := strings.Split(b.netMapCache.Concise(), "\n")
s2 := strings.Split(new.NetMap.Concise(), "\n") s2 := strings.Split(newSt.NetMap.Concise(), "\n")
b.logf("netmap diff:\n%v\n", b.cmpDiff(s1, s2)) b.logf("netmap diff:\n%v\n", b.cmpDiff(s1, s2))
} }
b.netMapCache = new.NetMap b.netMapCache = newSt.NetMap
b.send(Notify{NetMap: new.NetMap}) b.send(Notify{NetMap: newSt.NetMap})
b.updateFilter() b.updateFilter()
} }
if new.URL != "" { if newSt.URL != "" {
b.logf("Received auth URL: %.20v...\n", new.URL) b.logf("Received auth URL: %.20v...\n", newSt.URL)
b.mu.Lock() b.mu.Lock()
interact := b.interact interact := b.interact
b.authURL = new.URL b.authURL = newSt.URL
b.mu.Unlock() b.mu.Unlock()
if interact > 0 { if interact > 0 {
b.popBrowserAuthNow() b.popBrowserAuthNow()
} }
} }
if new.Err != "" { if newSt.Err != "" {
// TODO(crawshaw): display in the UI. // TODO(crawshaw): display in the UI.
log.Print(new.Err) log.Print(newSt.Err)
return return
} }
if new.NetMap != nil { if newSt.NetMap != nil {
if b.prefs.WantRunning || b.State() == NeedsLogin { if b.prefs.WantRunning || b.State() == NeedsLogin {
b.prefs.WantRunning = true b.prefs.WantRunning = true
} }
@ -255,14 +256,15 @@ func (b *LocalBackend) Start(opts Options) error {
log.Fatalf("weird: non-error wgengine update with status=nil\n") log.Fatalf("weird: non-error wgengine update with status=nil\n")
} }
b.mu.Lock() b.mu.Lock() // why does this hold b.mu? parseWgStatus only reads b.logf
es := b.parseWgStatus(s) es := b.parseWgStatus(s)
c := b.c
b.mu.Unlock() b.mu.Unlock()
b.engineStatus = es b.engineStatus = es
if b.c != nil { if c != nil {
b.c.UpdateEndpoints(0, s.LocalAddrs) c.UpdateEndpoints(0, s.LocalAddrs)
} }
b.endPoints = append([]string{}, s.LocalAddrs...) b.endPoints = append([]string{}, s.LocalAddrs...)
b.stateMachine() b.stateMachine()
@ -342,9 +344,13 @@ func (b *LocalBackend) runPoller() {
} }
func (b *LocalBackend) send(n Notify) { func (b *LocalBackend) send(n Notify) {
if b.notify != nil { b.mu.Lock()
notify := b.notify
b.mu.Unlock()
if notify != nil {
n.Version = version.LONG n.Version = version.LONG
b.notify(n) notify(n)
} }
} }
@ -364,7 +370,8 @@ func (b *LocalBackend) popBrowserAuthNow() {
} }
} }
func (b *LocalBackend) loadStateWithLock(key StateKey, prefs *Prefs, legacyPath string) error { // b.mu must be held
func (b *LocalBackend) loadStateLocked(key StateKey, prefs *Prefs, legacyPath string) error {
if prefs == nil && key == "" { if prefs == nil && key == "" {
panic("state key and prefs are both unset") panic("state key and prefs are both unset")
} }
@ -430,17 +437,18 @@ func (b *LocalBackend) EngineStatus() EngineStatus {
} }
func (b *LocalBackend) StartLoginInteractive() { func (b *LocalBackend) StartLoginInteractive() {
b.assertClient()
b.mu.Lock() b.mu.Lock()
b.assertClientLocked()
b.interact++ b.interact++
url := b.authURL url := b.authURL
c := b.c
b.mu.Unlock() b.mu.Unlock()
b.logf("StartLoginInteractive: url=%v\n", url != "") b.logf("StartLoginInteractive: url=%v\n", url != "")
if url != "" { if url != "" {
b.popBrowserAuthNow() b.popBrowserAuthNow()
} else { } else {
b.c.Login(nil, controlclient.LoginInteractive) c.Login(nil, controlclient.LoginInteractive)
} }
} }
@ -618,6 +626,7 @@ func (b *LocalBackend) enterState(newState State) {
b.mu.Lock() b.mu.Lock()
state := b.state state := b.state
prefs := b.prefs prefs := b.prefs
notify := b.notify
b.mu.Unlock() b.mu.Unlock()
if state == newState { if state == newState {
@ -625,7 +634,7 @@ func (b *LocalBackend) enterState(newState State) {
} }
b.logf("Switching ipn state %v -> %v (WantRunning=%v)\n", b.logf("Switching ipn state %v -> %v (WantRunning=%v)\n",
state, newState, prefs.WantRunning) state, newState, prefs.WantRunning)
if b.notify != nil { if notify != nil {
b.send(Notify{State: &newState}) b.send(Notify{State: &newState})
} }
@ -652,11 +661,11 @@ func (b *LocalBackend) enterState(newState State) {
} }
func (b *LocalBackend) nextState() State { func (b *LocalBackend) nextState() State {
b.assertClient() c := b.mustClient()
state := b.State() state := b.State()
if b.netMapCache == nil { if b.netMapCache == nil {
if b.c.AuthCantContinue() { if c.AuthCantContinue() {
// Auth was interrupted or waiting for URL visit, // Auth was interrupted or waiting for URL visit,
// so it won't proceed without human help. // so it won't proceed without human help.
return NeedsLogin return NeedsLogin
@ -721,14 +730,21 @@ func (b *LocalBackend) requestEngineStatusAndWait() {
// Maybe that's for the better; if someone logs out accidentally, // Maybe that's for the better; if someone logs out accidentally,
// rebooting will fix it. // rebooting will fix it.
func (b *LocalBackend) Logout() { func (b *LocalBackend) Logout() {
b.assertClient() c := b.mustClient()
b.netMapCache = nil b.netMapCache = nil
b.c.Logout() c.Logout()
b.netMapCache = nil b.netMapCache = nil
b.stateMachine() b.stateMachine()
} }
func (b *LocalBackend) assertClient() { func (b *LocalBackend) mustClient() *controlclient.Client {
b.mu.Lock()
defer b.mu.Unlock()
b.assertClientLocked()
return b.c
}
func (b *LocalBackend) assertClientLocked() {
if b.c == nil { if b.c == nil {
panic("LocalBackend.assertClient: b.c == nil") panic("LocalBackend.assertClient: b.c == nil")
} }