htlcswitch/mailbox: advance packet head after delivery

This commit delays the advancement of the pktHead until after the
message has been delivered. This is a prepatory step, as in the future
we may fail to deliver the packet due to a deadline expiring.
This commit is contained in:
Conner Fromknecht 2020-04-14 10:48:58 -07:00
parent 564534c829
commit 63f3d0b012
No known key found for this signature in database
GPG Key ID: E7D737B67FA592C7

@ -219,6 +219,13 @@ func (m *memoryMailBox) AckPacket(inKey CircuitKey) bool {
return false return false
} }
// Check whether we are removing the head of the queue. If so, we must
// advance the head to the next packet before removing. It's possible
// that the courier has already adanced the pktHead, so this check
// prevents the pktHead from getting desynchronized.
if entry == m.pktHead {
m.pktHead = entry.Next()
}
m.htlcPkts.Remove(entry) m.htlcPkts.Remove(entry)
delete(m.pktIndex, inKey) delete(m.pktIndex, inKey)
m.pktCond.L.Unlock() m.pktCond.L.Unlock()
@ -334,6 +341,7 @@ func (m *memoryMailBox) mailCourier(cType courierType) {
var ( var (
nextPkt *htlcPacket nextPkt *htlcPacket
nextPktEl *list.Element
nextMsg lnwire.Message nextMsg lnwire.Message
) )
switch cType { switch cType {
@ -350,7 +358,7 @@ func (m *memoryMailBox) mailCourier(cType courierType) {
// re-delivered once the link comes back online. // re-delivered once the link comes back online.
case pktCourier: case pktCourier:
nextPkt = m.pktHead.Value.(*htlcPacket) nextPkt = m.pktHead.Value.(*htlcPacket)
m.pktHead = m.pktHead.Next() nextPktEl = m.pktHead
} }
// Now that we're done with the condition, we can unlock it to // Now that we're done with the condition, we can unlock it to
@ -382,6 +390,14 @@ func (m *memoryMailBox) mailCourier(cType courierType) {
case pktCourier: case pktCourier:
select { select {
case m.pktOutbox <- nextPkt: case m.pktOutbox <- nextPkt:
m.pktCond.L.Lock()
// Only advance the pktHead if this packet
// is still at the head of the queue.
if m.pktHead != nil && m.pktHead == nextPktEl {
m.pktHead = m.pktHead.Next()
}
m.pktCond.L.Unlock()
case pktDone := <-m.pktReset: case pktDone := <-m.pktReset:
m.pktCond.L.Lock() m.pktCond.L.Lock()
m.pktHead = m.htlcPkts.Front() m.pktHead = m.htlcPkts.Front()