htlcswithc/mailbox: prioritize settles/fails over adds
This commit splits the packet courier internally into two distinct queues, one for adds and one for settles+fails. This allows us to prioritize HTLCs that will clear the commitment transaction and make space for adds. Previously this responsibility was handled by the overflow queue.
This commit is contained in:
parent
37dca27a3d
commit
1aa2dde4a4
@ -4,6 +4,7 @@ import (
|
|||||||
"bytes"
|
"bytes"
|
||||||
"container/list"
|
"container/list"
|
||||||
"errors"
|
"errors"
|
||||||
|
"fmt"
|
||||||
"sync"
|
"sync"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
@ -108,6 +109,11 @@ type memoryMailBox struct {
|
|||||||
htlcPkts *list.List
|
htlcPkts *list.List
|
||||||
pktIndex map[CircuitKey]*list.Element
|
pktIndex map[CircuitKey]*list.Element
|
||||||
pktHead *list.Element
|
pktHead *list.Element
|
||||||
|
|
||||||
|
addPkts *list.List
|
||||||
|
addIndex map[CircuitKey]*list.Element
|
||||||
|
addHead *list.Element
|
||||||
|
|
||||||
pktMtx sync.Mutex
|
pktMtx sync.Mutex
|
||||||
pktCond *sync.Cond
|
pktCond *sync.Cond
|
||||||
|
|
||||||
@ -125,11 +131,13 @@ func newMemoryMailBox(cfg *mailBoxConfig) *memoryMailBox {
|
|||||||
cfg: cfg,
|
cfg: cfg,
|
||||||
wireMessages: list.New(),
|
wireMessages: list.New(),
|
||||||
htlcPkts: list.New(),
|
htlcPkts: list.New(),
|
||||||
|
addPkts: list.New(),
|
||||||
messageOutbox: make(chan lnwire.Message),
|
messageOutbox: make(chan lnwire.Message),
|
||||||
pktOutbox: make(chan *htlcPacket),
|
pktOutbox: make(chan *htlcPacket),
|
||||||
msgReset: make(chan chan struct{}, 1),
|
msgReset: make(chan chan struct{}, 1),
|
||||||
pktReset: make(chan chan struct{}, 1),
|
pktReset: make(chan chan struct{}, 1),
|
||||||
pktIndex: make(map[CircuitKey]*list.Element),
|
pktIndex: make(map[CircuitKey]*list.Element),
|
||||||
|
addIndex: make(map[CircuitKey]*list.Element),
|
||||||
wireShutdown: make(chan struct{}),
|
wireShutdown: make(chan struct{}),
|
||||||
pktShutdown: make(chan struct{}),
|
pktShutdown: make(chan struct{}),
|
||||||
quit: make(chan struct{}),
|
quit: make(chan struct{}),
|
||||||
@ -222,26 +230,41 @@ func (m *memoryMailBox) signalUntilReset(cType courierType,
|
|||||||
// NOTE: It is safe to call this method multiple times for the same circuit key.
|
// NOTE: It is safe to call this method multiple times for the same circuit key.
|
||||||
func (m *memoryMailBox) AckPacket(inKey CircuitKey) bool {
|
func (m *memoryMailBox) AckPacket(inKey CircuitKey) bool {
|
||||||
m.pktCond.L.Lock()
|
m.pktCond.L.Lock()
|
||||||
entry, ok := m.pktIndex[inKey]
|
defer m.pktCond.L.Unlock()
|
||||||
if !ok {
|
|
||||||
m.pktCond.L.Unlock()
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
// Check whether we are removing the head of the queue. If so, we must
|
if entry, ok := m.pktIndex[inKey]; ok {
|
||||||
// advance the head to the next packet before removing. It's possible
|
// Check whether we are removing the head of the queue. If so,
|
||||||
// that the courier has already adanced the pktHead, so this check
|
// we must advance the head to the next packet before removing.
|
||||||
// prevents the pktHead from getting desynchronized.
|
// It's possible that the courier has already advanced the
|
||||||
|
// pktHead, so this check prevents the pktHead from getting
|
||||||
|
// desynchronized.
|
||||||
if entry == m.pktHead {
|
if entry == m.pktHead {
|
||||||
m.pktHead = entry.Next()
|
m.pktHead = entry.Next()
|
||||||
}
|
}
|
||||||
m.htlcPkts.Remove(entry)
|
m.htlcPkts.Remove(entry)
|
||||||
delete(m.pktIndex, inKey)
|
delete(m.pktIndex, inKey)
|
||||||
m.pktCond.L.Unlock()
|
|
||||||
|
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if entry, ok := m.addIndex[inKey]; ok {
|
||||||
|
// Check whether we are removing the head of the queue. If so,
|
||||||
|
// we must advance the head to the next add before removing.
|
||||||
|
// It's possible that the courier has already advanced the
|
||||||
|
// addHead, so this check prevents the addHead from getting
|
||||||
|
// desynchronized.
|
||||||
|
if entry == m.addHead {
|
||||||
|
m.addHead = entry.Next()
|
||||||
|
}
|
||||||
|
m.addPkts.Remove(entry)
|
||||||
|
delete(m.addIndex, inKey)
|
||||||
|
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
// HasPacket queries the packets for a circuit key, this is used to drop packets
|
// HasPacket queries the packets for a circuit key, this is used to drop packets
|
||||||
// bound for the switch that already have a queued response.
|
// bound for the switch that already have a queued response.
|
||||||
func (m *memoryMailBox) HasPacket(inKey CircuitKey) bool {
|
func (m *memoryMailBox) HasPacket(inKey CircuitKey) bool {
|
||||||
@ -328,7 +351,7 @@ func (m *memoryMailBox) mailCourier(cType courierType) {
|
|||||||
|
|
||||||
case pktCourier:
|
case pktCourier:
|
||||||
m.pktCond.L.Lock()
|
m.pktCond.L.Lock()
|
||||||
for m.pktHead == nil {
|
for m.pktHead == nil && m.addHead == nil {
|
||||||
m.pktCond.Wait()
|
m.pktCond.Wait()
|
||||||
|
|
||||||
select {
|
select {
|
||||||
@ -338,6 +361,7 @@ func (m *memoryMailBox) mailCourier(cType courierType) {
|
|||||||
// reconnect.
|
// reconnect.
|
||||||
case pktDone := <-m.pktReset:
|
case pktDone := <-m.pktReset:
|
||||||
m.pktHead = m.htlcPkts.Front()
|
m.pktHead = m.htlcPkts.Front()
|
||||||
|
m.addHead = m.addPkts.Front()
|
||||||
|
|
||||||
close(pktDone)
|
close(pktDone)
|
||||||
case <-m.quit:
|
case <-m.quit:
|
||||||
@ -351,6 +375,8 @@ func (m *memoryMailBox) mailCourier(cType courierType) {
|
|||||||
var (
|
var (
|
||||||
nextPkt *htlcPacket
|
nextPkt *htlcPacket
|
||||||
nextPktEl *list.Element
|
nextPktEl *list.Element
|
||||||
|
nextAdd *htlcPacket
|
||||||
|
nextAddEl *list.Element
|
||||||
nextMsg lnwire.Message
|
nextMsg lnwire.Message
|
||||||
)
|
)
|
||||||
switch cType {
|
switch cType {
|
||||||
@ -366,8 +392,15 @@ func (m *memoryMailBox) mailCourier(cType courierType) {
|
|||||||
// doesn't make it into a commitment, then it'll be
|
// doesn't make it into a commitment, then it'll be
|
||||||
// re-delivered once the link comes back online.
|
// re-delivered once the link comes back online.
|
||||||
case pktCourier:
|
case pktCourier:
|
||||||
|
// Peek at the next item to deliver, prioritizing
|
||||||
|
// Settle/Fail packets over Adds.
|
||||||
|
if m.pktHead != nil {
|
||||||
nextPkt = m.pktHead.Value.(*htlcPacket)
|
nextPkt = m.pktHead.Value.(*htlcPacket)
|
||||||
nextPktEl = m.pktHead
|
nextPktEl = m.pktHead
|
||||||
|
} else {
|
||||||
|
nextAdd = m.addHead.Value.(*htlcPacket)
|
||||||
|
nextAddEl = m.addHead
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// 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
|
||||||
@ -397,22 +430,56 @@ func (m *memoryMailBox) mailCourier(cType courierType) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
case pktCourier:
|
case pktCourier:
|
||||||
|
var (
|
||||||
|
pktOutbox chan *htlcPacket
|
||||||
|
addOutbox chan *htlcPacket
|
||||||
|
)
|
||||||
|
|
||||||
|
// Prioritize delivery of Settle/Fail packets over Adds.
|
||||||
|
// This ensures that we actively clear the commitment of
|
||||||
|
// existing HTLCs before trying to add new ones. This
|
||||||
|
// can help to improve forwarding performance since the
|
||||||
|
// time to sign a commitment is linear in the number of
|
||||||
|
// HTLCs manifested on the commitments.
|
||||||
|
//
|
||||||
|
// NOTE: Both types are eventually delivered over the
|
||||||
|
// same channel, but we can control which is delivered
|
||||||
|
// by exclusively making one nil and the other non-nil.
|
||||||
|
// We know from our loop condition that at least one
|
||||||
|
// nextPkt and nextAdd are non-nil.
|
||||||
|
if nextPkt != nil {
|
||||||
|
pktOutbox = m.pktOutbox
|
||||||
|
} else {
|
||||||
|
addOutbox = m.pktOutbox
|
||||||
|
}
|
||||||
|
|
||||||
select {
|
select {
|
||||||
case m.pktOutbox <- nextPkt:
|
case pktOutbox <- nextPkt:
|
||||||
m.pktCond.L.Lock()
|
m.pktCond.L.Lock()
|
||||||
// Only advance the pktHead if this packet
|
// Only advance the pktHead if this Settle or
|
||||||
// is still at the head of the queue.
|
// Fail is still at the head of the queue.
|
||||||
if m.pktHead != nil && m.pktHead == nextPktEl {
|
if m.pktHead != nil && m.pktHead == nextPktEl {
|
||||||
m.pktHead = m.pktHead.Next()
|
m.pktHead = m.pktHead.Next()
|
||||||
}
|
}
|
||||||
m.pktCond.L.Unlock()
|
m.pktCond.L.Unlock()
|
||||||
|
|
||||||
|
case addOutbox <- nextAdd:
|
||||||
|
m.pktCond.L.Lock()
|
||||||
|
// Only advance the addHead if this Add is still
|
||||||
|
// at the head of the queue.
|
||||||
|
if m.addHead != nil && m.addHead == nextAddEl {
|
||||||
|
m.addHead = m.addHead.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()
|
||||||
|
m.addHead = m.addPkts.Front()
|
||||||
m.pktCond.L.Unlock()
|
m.pktCond.L.Unlock()
|
||||||
|
|
||||||
close(pktDone)
|
close(pktDone)
|
||||||
|
|
||||||
case <-m.quit:
|
case <-m.quit:
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
@ -444,9 +511,11 @@ func (m *memoryMailBox) AddMessage(msg lnwire.Message) error {
|
|||||||
// NOTE: This method is safe for concrete use and part of the MailBox
|
// NOTE: This method is safe for concrete use and part of the MailBox
|
||||||
// interface.
|
// interface.
|
||||||
func (m *memoryMailBox) AddPacket(pkt *htlcPacket) error {
|
func (m *memoryMailBox) AddPacket(pkt *htlcPacket) error {
|
||||||
// First, we'll lock the condition, and add the packet to the end of
|
|
||||||
// the htlc packet inbox.
|
|
||||||
m.pktCond.L.Lock()
|
m.pktCond.L.Lock()
|
||||||
|
switch htlc := pkt.htlc.(type) {
|
||||||
|
|
||||||
|
// Split off Settle/Fail packets into the htlcPkts queue.
|
||||||
|
case *lnwire.UpdateFulfillHTLC, *lnwire.UpdateFailHTLC:
|
||||||
if _, ok := m.pktIndex[pkt.inKey()]; ok {
|
if _, ok := m.pktIndex[pkt.inKey()]; ok {
|
||||||
m.pktCond.L.Unlock()
|
m.pktCond.L.Unlock()
|
||||||
return nil
|
return nil
|
||||||
@ -457,6 +526,24 @@ func (m *memoryMailBox) AddPacket(pkt *htlcPacket) error {
|
|||||||
if m.pktHead == nil {
|
if m.pktHead == nil {
|
||||||
m.pktHead = entry
|
m.pktHead = entry
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Split off Add packets into the addPkts queue.
|
||||||
|
case *lnwire.UpdateAddHTLC:
|
||||||
|
if _, ok := m.addIndex[pkt.inKey()]; ok {
|
||||||
|
m.pktCond.L.Unlock()
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
entry := m.addPkts.PushBack(pkt)
|
||||||
|
m.addIndex[pkt.inKey()] = entry
|
||||||
|
if m.addHead == nil {
|
||||||
|
m.addHead = entry
|
||||||
|
}
|
||||||
|
|
||||||
|
default:
|
||||||
|
m.pktCond.L.Unlock()
|
||||||
|
return fmt.Errorf("unknown htlc type: %T", htlc)
|
||||||
|
}
|
||||||
m.pktCond.L.Unlock()
|
m.pktCond.L.Unlock()
|
||||||
|
|
||||||
// With the packet added, we signal to the mailCourier that there are
|
// With the packet added, we signal to the mailCourier that there are
|
||||||
|
@ -38,6 +38,9 @@ func TestMailBoxCouriers(t *testing.T) {
|
|||||||
outgoingChanID: lnwire.NewShortChanIDFromInt(uint64(prand.Int63())),
|
outgoingChanID: lnwire.NewShortChanIDFromInt(uint64(prand.Int63())),
|
||||||
incomingChanID: lnwire.NewShortChanIDFromInt(uint64(prand.Int63())),
|
incomingChanID: lnwire.NewShortChanIDFromInt(uint64(prand.Int63())),
|
||||||
amount: lnwire.MilliSatoshi(prand.Int63()),
|
amount: lnwire.MilliSatoshi(prand.Int63()),
|
||||||
|
htlc: &lnwire.UpdateAddHTLC{
|
||||||
|
ID: uint64(i),
|
||||||
|
},
|
||||||
}
|
}
|
||||||
sentPackets[i] = pkt
|
sentPackets[i] = pkt
|
||||||
|
|
||||||
@ -315,6 +318,106 @@ func TestMailBoxFailAdd(t *testing.T) {
|
|||||||
// duplicate fails are sent.
|
// duplicate fails are sent.
|
||||||
go failAdds(adds)
|
go failAdds(adds)
|
||||||
ctx.checkFails(nil)
|
ctx.checkFails(nil)
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
// TestMailBoxPacketPrioritization asserts that the mailbox will prioritize
|
||||||
|
// delivering Settle and Fail packets over Adds if both are available for
|
||||||
|
// delivery at the same time.
|
||||||
|
func TestMailBoxPacketPrioritization(t *testing.T) {
|
||||||
|
t.Parallel()
|
||||||
|
|
||||||
|
// First, we'll create new instance of the current default mailbox
|
||||||
|
// type.
|
||||||
|
mailBox := newMemoryMailBox(&mailBoxConfig{
|
||||||
|
clock: clock.NewDefaultClock(),
|
||||||
|
expiry: time.Minute,
|
||||||
|
})
|
||||||
|
mailBox.Start()
|
||||||
|
defer mailBox.Stop()
|
||||||
|
|
||||||
|
const numPackets = 5
|
||||||
|
|
||||||
|
_, _, aliceChanID, bobChanID := genIDs()
|
||||||
|
|
||||||
|
// Next we'll send the following sequence of packets:
|
||||||
|
// - Settle1
|
||||||
|
// - Add1
|
||||||
|
// - Add2
|
||||||
|
// - Fail
|
||||||
|
// - Settle2
|
||||||
|
sentPackets := make([]*htlcPacket, numPackets)
|
||||||
|
for i := 0; i < numPackets; i++ {
|
||||||
|
pkt := &htlcPacket{
|
||||||
|
outgoingChanID: aliceChanID,
|
||||||
|
outgoingHTLCID: uint64(i),
|
||||||
|
incomingChanID: bobChanID,
|
||||||
|
incomingHTLCID: uint64(i),
|
||||||
|
amount: lnwire.MilliSatoshi(prand.Int63()),
|
||||||
|
}
|
||||||
|
|
||||||
|
switch i {
|
||||||
|
case 0, 4:
|
||||||
|
// First and last packets are a Settle. A non-Add is
|
||||||
|
// sent first to make the test deterministic w/o needing
|
||||||
|
// to sleep.
|
||||||
|
pkt.htlc = &lnwire.UpdateFulfillHTLC{ID: uint64(i)}
|
||||||
|
case 1, 2:
|
||||||
|
// Next two packets are Adds.
|
||||||
|
pkt.htlc = &lnwire.UpdateAddHTLC{ID: uint64(i)}
|
||||||
|
case 3:
|
||||||
|
// Last packet is a Fail.
|
||||||
|
pkt.htlc = &lnwire.UpdateFailHTLC{ID: uint64(i)}
|
||||||
|
}
|
||||||
|
|
||||||
|
sentPackets[i] = pkt
|
||||||
|
|
||||||
|
err := mailBox.AddPacket(pkt)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("failed to add packet: %v", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// When dequeueing the packets, we expect the following sequence:
|
||||||
|
// - Settle1
|
||||||
|
// - Fail
|
||||||
|
// - Settle2
|
||||||
|
// - Add1
|
||||||
|
// - Add2
|
||||||
|
//
|
||||||
|
// We expect to see Fail and Settle2 to be delivered before either Add1
|
||||||
|
// or Add2 due to the prioritization between the split queue.
|
||||||
|
for i := 0; i < numPackets; i++ {
|
||||||
|
select {
|
||||||
|
case pkt := <-mailBox.PacketOutBox():
|
||||||
|
var expPkt *htlcPacket
|
||||||
|
switch i {
|
||||||
|
case 0:
|
||||||
|
// First packet should be Settle1.
|
||||||
|
expPkt = sentPackets[0]
|
||||||
|
case 1:
|
||||||
|
// Second packet should be Fail.
|
||||||
|
expPkt = sentPackets[3]
|
||||||
|
case 2:
|
||||||
|
// Third packet should be Settle2.
|
||||||
|
expPkt = sentPackets[4]
|
||||||
|
case 3:
|
||||||
|
// Fourth packet should be Add1.
|
||||||
|
expPkt = sentPackets[1]
|
||||||
|
case 4:
|
||||||
|
// Last packet should be Add2.
|
||||||
|
expPkt = sentPackets[2]
|
||||||
|
}
|
||||||
|
|
||||||
|
if !reflect.DeepEqual(expPkt, pkt) {
|
||||||
|
t.Fatalf("recvd packet mismatch %d, want: %v, got: %v",
|
||||||
|
i, spew.Sdump(expPkt), spew.Sdump(pkt))
|
||||||
|
}
|
||||||
|
|
||||||
|
case <-time.After(50 * time.Millisecond):
|
||||||
|
t.Fatalf("didn't receive packet %d before timeout", i)
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// TestMailOrchestrator asserts that the orchestrator properly buffers packets
|
// TestMailOrchestrator asserts that the orchestrator properly buffers packets
|
||||||
@ -346,6 +449,9 @@ func TestMailOrchestrator(t *testing.T) {
|
|||||||
incomingChanID: bobChanID,
|
incomingChanID: bobChanID,
|
||||||
incomingHTLCID: uint64(i),
|
incomingHTLCID: uint64(i),
|
||||||
amount: lnwire.MilliSatoshi(prand.Int63()),
|
amount: lnwire.MilliSatoshi(prand.Int63()),
|
||||||
|
htlc: &lnwire.UpdateAddHTLC{
|
||||||
|
ID: uint64(i),
|
||||||
|
},
|
||||||
}
|
}
|
||||||
sentPackets[i] = pkt
|
sentPackets[i] = pkt
|
||||||
|
|
||||||
@ -411,6 +517,9 @@ func TestMailOrchestrator(t *testing.T) {
|
|||||||
incomingChanID: bobChanID,
|
incomingChanID: bobChanID,
|
||||||
incomingHTLCID: uint64(halfPackets + i),
|
incomingHTLCID: uint64(halfPackets + i),
|
||||||
amount: lnwire.MilliSatoshi(prand.Int63()),
|
amount: lnwire.MilliSatoshi(prand.Int63()),
|
||||||
|
htlc: &lnwire.UpdateAddHTLC{
|
||||||
|
ID: uint64(halfPackets + i),
|
||||||
|
},
|
||||||
}
|
}
|
||||||
sentPackets[i] = pkt
|
sentPackets[i] = pkt
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user