From 0b5afa64f37e5cbdfe5067ab058f3d9842413d21 Mon Sep 17 00:00:00 2001 From: Joost Jager Date: Wed, 10 Apr 2019 16:05:09 +0200 Subject: [PATCH] htlcswitch: remove logCommitTick Replace logCommitTick as a way to deal with revocation window exhaustion by retrying to update the commit tx when the remote revocation is received. The rationale is that the revocation window always opens up because of a revoke message that is received from the other party. It is therefore not necessary to set a timer for this. The reception of the revoke message is the trigger to send a new commit sig if necessary. --- htlcswitch/link.go | 75 ++++++++--------------------------------- htlcswitch/link_test.go | 14 ++------ 2 files changed, 17 insertions(+), 72 deletions(-) diff --git a/htlcswitch/link.go b/htlcswitch/link.go index a2f89bb3..83a9c70e 100644 --- a/htlcswitch/link.go +++ b/htlcswitch/link.go @@ -344,14 +344,6 @@ type channelLink struct { // sub-systems with the latest set of active HTLC's on our channel. htlcUpdates chan *contractcourt.ContractUpdate - // logCommitTimer is a timer which is sent upon if we go an interval - // without receiving/sending a commitment update. It's role is to - // ensure both chains converge to identical state in a timely manner. - // - // TODO(roasbeef): timer should be >> then RTT - logCommitTimer *time.Timer - logCommitTick <-chan time.Time - // updateFeeTimer is the timer responsible for updating the link's // commitment fee every time it fires. updateFeeTimer *time.Timer @@ -397,13 +389,12 @@ func NewChannelLink(cfg ChannelLinkConfig, channel: channel, shortChanID: channel.ShortChanID(), // TODO(roasbeef): just do reserve here? - logCommitTimer: time.NewTimer(300 * time.Millisecond), - overflowQueue: newPacketQueue(input.MaxHTLCNumber / 2), - htlcUpdates: make(chan *contractcourt.ContractUpdate), - hodlMap: make(map[channeldb.CircuitKey]hodlHtlc), - hodlQueue: queue.NewConcurrentQueue(10), - log: build.NewPrefixLog(logPrefix, log), - quit: make(chan struct{}), + overflowQueue: newPacketQueue(input.MaxHTLCNumber / 2), + htlcUpdates: make(chan *contractcourt.ContractUpdate), + hodlMap: make(map[channeldb.CircuitKey]hodlHtlc), + hodlQueue: queue.NewConcurrentQueue(10), + log: build.NewPrefixLog(logPrefix, log), + quit: make(chan struct{}), } } @@ -1087,21 +1078,6 @@ out: break out - case <-l.logCommitTick: - // If we haven't sent or received a new commitment - // update in some time, check to see if we have any - // pending updates we need to commit due to our - // commitment chains being desynchronized. - if !l.channel.OweCommitment(true) { - continue - } - - if err := l.updateCommitTx(); err != nil { - l.fail(LinkFailureError{code: ErrInternalError}, - "unable to update commitment: %v", err) - break out - } - case <-l.cfg.BatchTicker.Ticks(): // Attempt to extend the remote commitment chain // including all the currently pending entries. If the @@ -1769,21 +1745,6 @@ func (l *channelLink) handleUpstreamMsg(msg lnwire.Message) { return } - // As we've just received a commitment signature, we'll - // re-start the log commit timer to wake up the main processing - // loop to check if we need to send a commitment signature as - // we owe one. - // - // TODO(roasbeef): instead after revocation? - if !l.logCommitTimer.Stop() { - select { - case <-l.logCommitTimer.C: - default: - } - } - l.logCommitTimer.Reset(300 * time.Millisecond) - l.logCommitTick = l.logCommitTimer.C - // If both commitment chains are fully synced from our PoV, // then we don't need to reply with a signature as both sides // already have a commitment with the latest accepted. @@ -1861,11 +1822,14 @@ func (l *channelLink) handleUpstreamMsg(msg lnwire.Message) { return } - // If there are pending local updates, try to update the commit - // tx. Pending updates could already have been present because - // of a previously failed update to the commit tx or freshly - // added by processRemoteAdds. - if l.channel.PendingLocalUpdateCount() > 0 { + // The revocation window opened up. If there are pending local + // updates, try to update the commit tx. Pending updates could + // already have been present because of a previously failed + // update to the commit tx or freshly added in by + // processRemoteAdds. Also in case there are no local updates, + // but there are still remote updates that are not in the remote + // commit tx yet, send out an update. + if l.channel.OweCommitment(true) { if err := l.updateCommitTx(); err != nil { l.fail(LinkFailureError{code: ErrInternalError}, "unable to update commitment: %v", err) @@ -2016,17 +1980,6 @@ func (l *channelLink) updateCommitTx() error { } l.cfg.Peer.SendMessage(false, commitSig) - // We've just initiated a state transition, attempt to stop the - // logCommitTimer. If the timer already ticked, then we'll consume the - // value, dropping - if l.logCommitTimer != nil && !l.logCommitTimer.Stop() { - select { - case <-l.logCommitTimer.C: - default: - } - } - l.logCommitTick = nil - return nil } diff --git a/htlcswitch/link_test.go b/htlcswitch/link_test.go index acbc4506..7d870fc0 100644 --- a/htlcswitch/link_test.go +++ b/htlcswitch/link_test.go @@ -5983,20 +5983,12 @@ func TestChannelLinkRevocationWindowRegular(t *testing.T) { // At this point, Alice cannot send a new commit sig to bob because the // revocation window is exhausted. - // Sleep to let the commit ticker expire. The revocation window is still - // exhausted. - time.Sleep(time.Second) - // Bob sends revocation and signs commit with htlc1 settled. ctx.sendRevAndAckBobToAlice() - // Allow some time for the log commit ticker to trigger for Alice. - time.Sleep(time.Second) - - // Now that Bob revoked, Alice should send the sig she owes. - // - // THIS SHOULD NOT HAPPEN. - ctx.assertNoMsgFromAlice(time.Second) + // After the revocation, it is again possible for Alice to send a commit + // sig with htlc2. + ctx.receiveCommitSigAliceToBob(1) } // TestChannelLinkRevocationWindowHodl asserts that htlcs paying to a hodl