htlcswitch: allow re-use of onion circuits via reference coutning
This commit fixes a bug in the switch which would manifest only in the case of a multi-hop payment which _reused_ the same payment hash over several distinct payments. In this case, after the first few payments were settled, the circuit would be deleted, meaning that once the remainder of the payment were forwarded, the switch wouldn’t know to _whom_ to settle them to. We fix this issue by using the refCount field to only garbage collect a circuit after there are no more references.
This commit is contained in:
parent
e7420edd44
commit
38ef8f98e2
@ -157,10 +157,6 @@ type circuitKey [32]byte
|
|||||||
// re-used (not torndown) in the case that multiple HTLCs with the send RHash
|
// re-used (not torndown) in the case that multiple HTLCs with the send RHash
|
||||||
// are sent.
|
// are sent.
|
||||||
type paymentCircuit struct {
|
type paymentCircuit struct {
|
||||||
// TODO(roasbeef): add reference count so know when to delete?
|
|
||||||
// * atomic int re
|
|
||||||
// * due to same r-value being re-used?
|
|
||||||
|
|
||||||
refCount uint32
|
refCount uint32
|
||||||
|
|
||||||
// clear is the link the htlcSwitch will forward the HTLC add message
|
// clear is the link the htlcSwitch will forward the HTLC add message
|
||||||
@ -372,6 +368,9 @@ out:
|
|||||||
deltaNumUpdates++
|
deltaNumUpdates++
|
||||||
|
|
||||||
hswcLog.Tracef("plex packet: %v", newLogClosure(func() string {
|
hswcLog.Tracef("plex packet: %v", newLogClosure(func() string {
|
||||||
|
if pkt.onion != nil {
|
||||||
|
pkt.onion.Packet.Header.EphemeralKey.Curve = nil
|
||||||
|
}
|
||||||
return spew.Sdump(pkt)
|
return spew.Sdump(pkt)
|
||||||
}))
|
}))
|
||||||
|
|
||||||
@ -453,18 +452,38 @@ out:
|
|||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
circuit := &paymentCircuit{
|
// Examine the circuit map to see if this
|
||||||
clear: clearLink[0],
|
// circuit is already in use or not. If so,
|
||||||
settle: settleLink,
|
// then we'll simply increment the reference
|
||||||
}
|
// count. Otherwise, we'll create a new circuit
|
||||||
|
// from scratch.
|
||||||
|
//
|
||||||
|
// TODO(roasbeef): include dest+src+amt in key
|
||||||
cKey := circuitKey(wireMsg.PaymentHash)
|
cKey := circuitKey(wireMsg.PaymentHash)
|
||||||
h.paymentCircuits[cKey] = circuit
|
circuit, ok := h.paymentCircuits[cKey]
|
||||||
|
if ok {
|
||||||
|
hswcLog.Debugf("Increasing ref_count "+
|
||||||
|
"of circuit: %x, from %v to %v",
|
||||||
|
wireMsg.PaymentHash,
|
||||||
|
circuit.refCount,
|
||||||
|
circuit.refCount+1)
|
||||||
|
|
||||||
hswcLog.Debugf("Creating onion circuit for %x: %v<->%v",
|
circuit.refCount += 1
|
||||||
|
} else {
|
||||||
|
hswcLog.Debugf("Creating onion "+
|
||||||
|
"circuit for %x: %v<->%v",
|
||||||
cKey[:], clearLink[0].chanID,
|
cKey[:], clearLink[0].chanID,
|
||||||
settleLink.chanID)
|
settleLink.chanID)
|
||||||
|
|
||||||
|
circuit = &paymentCircuit{
|
||||||
|
clear: clearLink[0],
|
||||||
|
settle: settleLink,
|
||||||
|
refCount: 1,
|
||||||
|
}
|
||||||
|
|
||||||
|
h.paymentCircuits[cKey] = circuit
|
||||||
|
}
|
||||||
|
|
||||||
// With the circuit initiated, send the htlcPkt
|
// With the circuit initiated, send the htlcPkt
|
||||||
// to the clearing link within the circuit to
|
// to the clearing link within the circuit to
|
||||||
// continue propagating the HTLC across the
|
// continue propagating the HTLC across the
|
||||||
@ -508,11 +527,6 @@ out:
|
|||||||
|
|
||||||
circuit.clear.restoreSlot()
|
circuit.clear.restoreSlot()
|
||||||
|
|
||||||
hswcLog.Debugf("Closing completed onion "+
|
|
||||||
"circuit for %x: %v<->%v", rHash[:],
|
|
||||||
circuit.clear.chanID,
|
|
||||||
circuit.settle.chanID)
|
|
||||||
|
|
||||||
circuit.settle.sendAndRestore(&htlcPacket{
|
circuit.settle.sendAndRestore(&htlcPacket{
|
||||||
msg: wireMsg,
|
msg: wireMsg,
|
||||||
err: make(chan error, 1),
|
err: make(chan error, 1),
|
||||||
@ -529,7 +543,13 @@ out:
|
|||||||
|
|
||||||
deltaSatSent += pkt.amt
|
deltaSatSent += pkt.amt
|
||||||
|
|
||||||
|
if circuit.refCount--; circuit.refCount == 0 {
|
||||||
|
hswcLog.Debugf("Closing completed onion "+
|
||||||
|
"circuit for %x: %v<->%v", rHash[:],
|
||||||
|
circuit.clear.chanID,
|
||||||
|
circuit.settle.chanID)
|
||||||
delete(h.paymentCircuits, cKey)
|
delete(h.paymentCircuits, cKey)
|
||||||
|
}
|
||||||
|
|
||||||
// We've just received an HTLC cancellation triggered
|
// We've just received an HTLC cancellation triggered
|
||||||
// by an upstream peer somewhere within the ultimate
|
// by an upstream peer somewhere within the ultimate
|
||||||
@ -568,8 +588,14 @@ out:
|
|||||||
err: make(chan error, 1),
|
err: make(chan error, 1),
|
||||||
})
|
})
|
||||||
|
|
||||||
|
if circuit.refCount--; circuit.refCount == 0 {
|
||||||
|
hswcLog.Debugf("Closing cancelled onion "+
|
||||||
|
"circuit for %x: %v<->%v", pkt.payHash,
|
||||||
|
circuit.clear.chanID,
|
||||||
|
circuit.settle.chanID)
|
||||||
delete(h.paymentCircuits, pkt.payHash)
|
delete(h.paymentCircuits, pkt.payHash)
|
||||||
}
|
}
|
||||||
|
}
|
||||||
case <-logTicker.C:
|
case <-logTicker.C:
|
||||||
if deltaNumUpdates == 0 {
|
if deltaNumUpdates == 0 {
|
||||||
continue
|
continue
|
||||||
|
Loading…
Reference in New Issue
Block a user