consensus: wait until peerUpdates channel is closed to close remaining peers

This commit is contained in:
William Banfield
2021-10-04 17:16:42 -04:00
parent b30ec89ee9
commit 2794e40071

View File

@@ -229,8 +229,34 @@ func (r *Reactor) OnStop() {
r.state.Wait()
}
// Close closeCh to signal to all spawned goroutines to gracefully exit. All
// p2p Channels should execute Close().
close(r.closeCh)
// Close the StateChannel goroutine separately since it uses its own channel
// to signal closure.
close(r.stateCloseCh)
<-r.stateCh.Done()
// Wait for all p2p Channels to be closed before returning. This ensures we
// can easily reason about synchronization of all p2p Channels and ensure no
// panics will occur.
<-r.voteSetBitsCh.Done()
<-r.dataCh.Done()
<-r.voteCh.Done()
<-r.peerUpdates.Done()
peers := make(map[types.NodeID]*PeerState)
r.mtx.Lock()
peers := r.peers
// Here, we make a copy of the map of peer states.
// The code below proceeds to access entries from the map outside of a lock.
// Goroutines launched by other functions may be deleting from the map but should
// to it since we waited for the peerUpdates channel to complete.
// Each of the methods below is safe to perform concurrently on the peers.
for id, state := range r.peers {
peers[id] = state
}
r.mtx.Unlock()
// wait for all spawned peer goroutines to gracefully exit
@@ -240,23 +266,6 @@ func (r *Reactor) OnStop() {
for _, ps := range peers {
ps.broadcastWG.Wait()
}
// Close the StateChannel goroutine separately since it uses its own channel
// to signal closure.
close(r.stateCloseCh)
<-r.stateCh.Done()
// Close closeCh to signal to all spawned goroutines to gracefully exit. All
// p2p Channels should execute Close().
close(r.closeCh)
// Wait for all p2p Channels to be closed before returning. This ensures we
// can easily reason about synchronization of all p2p Channels and ensure no
// panics will occur.
<-r.voteSetBitsCh.Done()
<-r.dataCh.Done()
<-r.voteCh.Done()
<-r.peerUpdates.Done()
}
// SetEventBus sets the reactor's event bus.