From 625d57aea61f53b729c7986041c71b6c584e0b55 Mon Sep 17 00:00:00 2001 From: Olaoluwa Osuntokun Date: Wed, 7 Jun 2017 17:04:27 -0700 Subject: [PATCH] chainntnfs/btcdnotify: ensure block epoch coroutines exit before closing ntfn channel --- chainntnfs/btcdnotify/btcd.go | 24 ++++++++++++++++++++---- 1 file changed, 20 insertions(+), 4 deletions(-) diff --git a/chainntnfs/btcdnotify/btcd.go b/chainntnfs/btcdnotify/btcd.go index 3a7f7e8b..069d22e4 100644 --- a/chainntnfs/btcdnotify/btcd.go +++ b/chainntnfs/btcdnotify/btcd.go @@ -185,7 +185,6 @@ func (b *BtcdNotifier) Stop() error { } } for _, epochClient := range b.blockEpochClients { - close(epochClient.cancelChan) close(epochClient.epochChan) } @@ -261,7 +260,17 @@ out: chainntnfs.Log.Infof("Cancelling epoch "+ "notification, epoch_id=%v", msg.epochID) + // First, close the cancel channel for this + // specific client, and wait for the client to + // exit. close(b.blockEpochClients[msg.epochID].cancelChan) + b.blockEpochClients[msg.epochID].wg.Wait() + + // Once the client has exited, we can then + // safely close the channel used to send epoch + // notifications, in order to notify any + // listeners that the intent has been + // cancelled. close(b.blockEpochClients[msg.epochID].epochChan) delete(b.blockEpochClients, msg.epochID) @@ -486,7 +495,11 @@ func (b *BtcdNotifier) notifyBlockEpochs(newHeight int32, newSha *chainhash.Hash for _, epochClient := range b.blockEpochClients { b.wg.Add(1) - go func(ntfnChan chan *chainntnfs.BlockEpoch, cancelChan chan struct{}) { + epochClient.wg.Add(1) + go func(ntfnChan chan *chainntnfs.BlockEpoch, cancelChan chan struct{}, + clientWg sync.WaitGroup) { + + defer clientWg.Done() defer b.wg.Done() select { @@ -498,7 +511,8 @@ func (b *BtcdNotifier) notifyBlockEpochs(newHeight int32, newSha *chainhash.Hash case <-b.quit: return } - }(epochClient.epochChan, epochClient.cancelChan) + + }(epochClient.epochChan, epochClient.cancelChan, epochClient.wg) } } @@ -712,11 +726,13 @@ func (b *BtcdNotifier) RegisterConfirmationsNtfn(txid *chainhash.Hash, // blockEpochRegistration represents a client's intent to receive a // notification with each newly connected block. type blockEpochRegistration struct { + epochID uint64 + epochChan chan *chainntnfs.BlockEpoch cancelChan chan struct{} - epochID uint64 + wg sync.WaitGroup } // epochCancel is a message sent to the BtcdNotifier when a client wishes to