Skip to content

Commit 19e8b5c

Browse files
committed
Consider funding scopes in get_available_balances
A FundedChannel may have more than one pending FundingScope during splicing, one for the splice attempt and one or more for any RBF attempts. When calling get_available_balances, consider all funding scopes and take the minimum by next_outbound_htlc_limit_msat. This is used both informationally and to determine which channel to use to forward an HTLC. The choice of next_outbound_htlc_limit_msat is somewhat arbitrary but matches the field used when determining which channel used to forward an HTLC. Any field should do since each field should be adjusted by the same amount relative to another FundingScope given the nature of the fields (i.e., inbound/outbound capacity, min/max HTLC limit). Using the minimum was chosen since an order for an HTLC to be sent over the channel, it must be possible for each funding scope -- both the confirmed one and any pending scopes, one of which may eventually confirm.
1 parent b6013f4 commit 19e8b5c

File tree

3 files changed

+67
-24
lines changed

3 files changed

+67
-24
lines changed

lightning/src/ln/channel.rs

+41-6
Original file line numberDiff line numberDiff line change
@@ -1550,6 +1550,25 @@ impl<SP: Deref> Channel<SP> where
15501550
}
15511551
}
15521552
}
1553+
1554+
/// Get the available balances, see [`AvailableBalances`]'s fields for more info.
1555+
/// Doesn't bother handling the
1556+
/// if-we-removed-it-already-but-haven't-fully-resolved-they-can-still-send-an-inbound-HTLC
1557+
/// corner case properly.
1558+
pub fn get_available_balances<F: Deref>(
1559+
&self, fee_estimator: &LowerBoundedFeeEstimator<F>,
1560+
) -> AvailableBalances
1561+
where
1562+
F::Target: FeeEstimator,
1563+
{
1564+
match &self.phase {
1565+
ChannelPhase::Undefined => unreachable!(),
1566+
ChannelPhase::Funded(chan) => chan.get_available_balances(fee_estimator),
1567+
ChannelPhase::UnfundedOutboundV1(chan) => chan.context.get_available_balances_for_scope(&chan.funding, fee_estimator),
1568+
ChannelPhase::UnfundedInboundV1(chan) => chan.context.get_available_balances_for_scope(&chan.funding, fee_estimator),
1569+
ChannelPhase::UnfundedV2(chan) => chan.context.get_available_balances_for_scope(&chan.funding, fee_estimator),
1570+
}
1571+
}
15531572
}
15541573

15551574
impl<SP: Deref> From<OutboundV1Channel<SP>> for Channel<SP>
@@ -4193,11 +4212,7 @@ impl<SP: Deref> ChannelContext<SP> where SP::Target: SignerProvider {
41934212
outbound_details
41944213
}
41954214

4196-
/// Get the available balances, see [`AvailableBalances`]'s fields for more info.
4197-
/// Doesn't bother handling the
4198-
/// if-we-removed-it-already-but-haven't-fully-resolved-they-can-still-send-an-inbound-HTLC
4199-
/// corner case properly.
4200-
pub fn get_available_balances<F: Deref>(
4215+
fn get_available_balances_for_scope<F: Deref>(
42014216
&self, funding: &FundingScope, fee_estimator: &LowerBoundedFeeEstimator<F>,
42024217
) -> AvailableBalances
42034218
where
@@ -8783,7 +8798,7 @@ impl<SP: Deref> FundedChannel<SP> where
87838798
return Err(ChannelError::Ignore("Cannot send 0-msat HTLC".to_owned()));
87848799
}
87858800

8786-
let available_balances = self.context.get_available_balances(&self.funding, fee_estimator);
8801+
let available_balances = self.get_available_balances(fee_estimator);
87878802
if amount_msat < available_balances.next_outbound_htlc_minimum_msat {
87888803
return Err(ChannelError::Ignore(format!("Cannot send less than our next-HTLC minimum - {} msat",
87898804
available_balances.next_outbound_htlc_minimum_msat)));
@@ -8855,6 +8870,26 @@ impl<SP: Deref> FundedChannel<SP> where
88558870
Ok(Some(res))
88568871
}
88578872

8873+
pub(super) fn get_available_balances<F: Deref>(
8874+
&self, fee_estimator: &LowerBoundedFeeEstimator<F>,
8875+
) -> AvailableBalances
8876+
where
8877+
F::Target: FeeEstimator,
8878+
{
8879+
core::iter::once(&self.funding)
8880+
.chain(self.pending_funding.iter())
8881+
.map(|funding| self.context.get_available_balances_for_scope(funding, fee_estimator))
8882+
.reduce(|acc, e| {
8883+
AvailableBalances {
8884+
inbound_capacity_msat: acc.inbound_capacity_msat.min(e.inbound_capacity_msat),
8885+
outbound_capacity_msat: acc.outbound_capacity_msat.min(e.outbound_capacity_msat),
8886+
next_outbound_htlc_limit_msat: acc.next_outbound_htlc_limit_msat.min(e.next_outbound_htlc_limit_msat),
8887+
next_outbound_htlc_minimum_msat: acc.next_outbound_htlc_minimum_msat.max(e.next_outbound_htlc_minimum_msat),
8888+
}
8889+
})
8890+
.expect("At least one FundingScope is always provided")
8891+
}
8892+
88588893
fn build_commitment_no_status_check<L: Deref>(&mut self, logger: &L) -> ChannelMonitorUpdate where L::Target: Logger {
88598894
log_trace!(logger, "Updating HTLC state for a newly-sent commitment_signed...");
88608895
// We can upgrade the status of some HTLCs that are waiting on a commitment, even if we

lightning/src/ln/channel_state.rs

+7-5
Original file line numberDiff line numberDiff line change
@@ -15,7 +15,7 @@ use bitcoin::secp256k1::PublicKey;
1515

1616
use crate::chain::chaininterface::{FeeEstimator, LowerBoundedFeeEstimator};
1717
use crate::chain::transaction::OutPoint;
18-
use crate::ln::channel::{ChannelContext, FundingScope};
18+
use crate::ln::channel::Channel;
1919
use crate::ln::types::ChannelId;
2020
use crate::sign::SignerProvider;
2121
use crate::types::features::{ChannelTypeFeatures, InitFeatures};
@@ -475,15 +475,17 @@ impl ChannelDetails {
475475
self.short_channel_id.or(self.outbound_scid_alias)
476476
}
477477

478-
pub(super) fn from_channel_context<SP: Deref, F: Deref>(
479-
context: &ChannelContext<SP>, funding: &FundingScope, best_block_height: u32,
480-
latest_features: InitFeatures, fee_estimator: &LowerBoundedFeeEstimator<F>,
478+
pub(super) fn from_channel<SP: Deref, F: Deref>(
479+
channel: &Channel<SP>, best_block_height: u32, latest_features: InitFeatures,
480+
fee_estimator: &LowerBoundedFeeEstimator<F>,
481481
) -> Self
482482
where
483483
SP::Target: SignerProvider,
484484
F::Target: FeeEstimator,
485485
{
486-
let balance = context.get_available_balances(funding, fee_estimator);
486+
let context = channel.context();
487+
let funding = channel.funding();
488+
let balance = channel.get_available_balances(fee_estimator);
487489
let (to_remote_reserve_satoshis, to_self_reserve_satoshis) =
488490
funding.get_holder_counterparty_selected_channel_reserve_satoshis();
489491
#[allow(deprecated)] // TODO: Remove once balance_msat is removed.

lightning/src/ln/channelmanager.rs

+19-13
Original file line numberDiff line numberDiff line change
@@ -3730,7 +3730,7 @@ where
37303730
Ok(temporary_channel_id)
37313731
}
37323732

3733-
fn list_funded_channels_with_filter<Fn: FnMut(&(&ChannelId, &FundedChannel<SP>)) -> bool + Copy>(&self, f: Fn) -> Vec<ChannelDetails> {
3733+
fn list_funded_channels_with_filter<Fn: FnMut(&(&ChannelId, &Channel<SP>)) -> bool + Copy>(&self, f: Fn) -> Vec<ChannelDetails> {
37343734
// Allocate our best estimate of the number of channels we have in the `res`
37353735
// Vec. Sadly the `short_to_chan_info` map doesn't cover channels without
37363736
// a scid or a scid alias. Therefore reallocations may still occur, but is
@@ -3745,11 +3745,13 @@ where
37453745
let peer_state = &mut *peer_state_lock;
37463746
res.extend(peer_state.channel_by_id.iter()
37473747
// Only `Channels` in the `Channel::Funded` phase can be considered funded.
3748-
.filter_map(|(chan_id, chan)| chan.as_funded().map(|chan| (chan_id, chan)))
3748+
.filter(|(_, chan)| chan.is_funded())
37493749
.filter(f)
37503750
.map(|(_channel_id, channel)| {
3751-
ChannelDetails::from_channel_context(&channel.context, &channel.funding, best_block_height,
3752-
peer_state.latest_features.clone(), &self.fee_estimator)
3751+
ChannelDetails::from_channel(
3752+
channel, best_block_height, peer_state.latest_features.clone(),
3753+
&self.fee_estimator,
3754+
)
37533755
})
37543756
);
37553757
}
@@ -3772,9 +3774,11 @@ where
37723774
for (_cp_id, peer_state_mutex) in per_peer_state.iter() {
37733775
let mut peer_state_lock = peer_state_mutex.lock().unwrap();
37743776
let peer_state = &mut *peer_state_lock;
3775-
for (context, funding) in peer_state.channel_by_id.iter().map(|(_, chan)| (chan.context(), chan.funding())) {
3776-
let details = ChannelDetails::from_channel_context(context, funding, best_block_height,
3777-
peer_state.latest_features.clone(), &self.fee_estimator);
3777+
for (_, channel) in peer_state.channel_by_id.iter() {
3778+
let details = ChannelDetails::from_channel(
3779+
channel, best_block_height, peer_state.latest_features.clone(),
3780+
&self.fee_estimator,
3781+
);
37783782
res.push(details);
37793783
}
37803784
}
@@ -3792,7 +3796,7 @@ where
37923796
// Note we use is_live here instead of usable which leads to somewhat confused
37933797
// internal/external nomenclature, but that's ok cause that's probably what the user
37943798
// really wanted anyway.
3795-
self.list_funded_channels_with_filter(|&(_, ref channel)| channel.context.is_live())
3799+
self.list_funded_channels_with_filter(|&(_, ref channel)| channel.context().is_live())
37963800
}
37973801

37983802
/// Gets the list of channels we have with a given counterparty, in random order.
@@ -3804,13 +3808,15 @@ where
38043808
let mut peer_state_lock = peer_state_mutex.lock().unwrap();
38053809
let peer_state = &mut *peer_state_lock;
38063810
let features = &peer_state.latest_features;
3807-
let context_to_details = |(context, funding)| {
3808-
ChannelDetails::from_channel_context(context, funding, best_block_height, features.clone(), &self.fee_estimator)
3811+
let channel_to_details = |channel| {
3812+
ChannelDetails::from_channel(
3813+
channel, best_block_height, features.clone(), &self.fee_estimator,
3814+
)
38093815
};
38103816
return peer_state.channel_by_id
38113817
.iter()
3812-
.map(|(_, chan)| (chan.context(), chan.funding()))
3813-
.map(context_to_details)
3818+
.map(|(_, chan)| (chan))
3819+
.map(channel_to_details)
38143820
.collect();
38153821
}
38163822
vec![]
@@ -6064,7 +6070,7 @@ where
60646070
let maybe_optimal_channel = peer_state.channel_by_id.values_mut()
60656071
.filter_map(Channel::as_funded_mut)
60666072
.filter_map(|chan| {
6067-
let balances = chan.context.get_available_balances(&chan.funding, &self.fee_estimator);
6073+
let balances = chan.get_available_balances(&self.fee_estimator);
60686074
if outgoing_amt_msat <= balances.next_outbound_htlc_limit_msat &&
60696075
outgoing_amt_msat >= balances.next_outbound_htlc_minimum_msat &&
60706076
chan.context.is_usable() {

0 commit comments

Comments
 (0)