2016-06-30 01:38:10 +02:00
|
|
|
#include "channel.h"
|
2016-07-01 03:57:57 +02:00
|
|
|
#include "htlc.h"
|
2016-08-18 06:53:46 +02:00
|
|
|
#include "remove_dust.h"
|
2017-01-04 04:37:15 +01:00
|
|
|
#include "type_to_string.h"
|
2015-07-29 08:44:28 +02:00
|
|
|
#include <assert.h>
|
2016-05-26 07:55:24 +02:00
|
|
|
#include <ccan/array_size/array_size.h>
|
2016-03-08 01:01:15 +01:00
|
|
|
#include <ccan/mem/mem.h>
|
2016-01-21 21:11:47 +01:00
|
|
|
#include <ccan/structeq/structeq.h>
|
2017-01-04 04:37:15 +01:00
|
|
|
#include <ccan/tal/str/str.h>
|
|
|
|
#include <inttypes.h>
|
2015-08-07 05:15:30 +02:00
|
|
|
#include <string.h>
|
2015-07-29 08:44:28 +02:00
|
|
|
|
2016-08-18 06:55:13 +02:00
|
|
|
uint64_t fee_by_feerate(size_t txsize, uint64_t fee_rate)
|
2016-03-24 02:42:43 +01:00
|
|
|
{
|
|
|
|
/* BOLT #2:
|
2016-11-11 00:02:04 +01:00
|
|
|
*
|
2016-05-02 08:28:56 +02:00
|
|
|
* The fee for a transaction MUST be calculated by multiplying this
|
|
|
|
* bytecount by the fee rate, dividing by 1000 and truncating
|
|
|
|
* (rounding down) the result to an even number of satoshis.
|
2016-03-24 02:42:43 +01:00
|
|
|
*/
|
|
|
|
return txsize * fee_rate / 2000 * 2;
|
|
|
|
}
|
2015-08-07 05:15:30 +02:00
|
|
|
|
2016-08-23 02:42:00 +02:00
|
|
|
/* BOLT #2:
|
|
|
|
*
|
|
|
|
* A node MUST use the formula 338 + 32 bytes for every non-dust HTLC
|
|
|
|
* as the bytecount for calculating commitment transaction fees. Note
|
|
|
|
* that the fee requirement is unchanged, even if the elimination of
|
|
|
|
* dust HTLC outputs has caused a non-zero fee already.
|
|
|
|
*/
|
|
|
|
static size_t tx_bytes(size_t num_nondust_htlcs)
|
|
|
|
{
|
|
|
|
return 338 + 32 * num_nondust_htlcs;
|
|
|
|
}
|
|
|
|
|
2016-03-24 02:42:43 +01:00
|
|
|
static uint64_t calculate_fee_msat(size_t num_nondust_htlcs,
|
2016-08-18 06:55:13 +02:00
|
|
|
uint64_t fee_rate)
|
2016-03-24 02:42:43 +01:00
|
|
|
{
|
|
|
|
/* milli-satoshis */
|
2016-08-23 02:42:00 +02:00
|
|
|
return fee_by_feerate(tx_bytes(num_nondust_htlcs), fee_rate) * 1000;
|
2015-08-07 05:15:30 +02:00
|
|
|
}
|
2015-08-07 05:15:30 +02:00
|
|
|
|
2016-03-24 02:42:43 +01:00
|
|
|
/* Pay this much fee, if possible. Return amount unpaid. */
|
|
|
|
static uint64_t pay_fee(struct channel_oneside *side, uint64_t fee_msat)
|
|
|
|
{
|
|
|
|
if (side->pay_msat >= fee_msat) {
|
|
|
|
side->pay_msat -= fee_msat;
|
|
|
|
side->fee_msat += fee_msat;
|
|
|
|
return 0;
|
|
|
|
} else {
|
|
|
|
uint64_t remainder = fee_msat - side->pay_msat;
|
|
|
|
side->fee_msat += side->pay_msat;
|
|
|
|
side->pay_msat = 0;
|
|
|
|
return remainder;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Charge the fee as per BOLT #2 */
|
|
|
|
static void recalculate_fees(struct channel_oneside *a,
|
|
|
|
struct channel_oneside *b,
|
|
|
|
uint64_t fee_msat)
|
|
|
|
{
|
|
|
|
uint64_t remainder;
|
|
|
|
|
|
|
|
/* Fold in fees, to recalcuate again below. */
|
|
|
|
a->pay_msat += a->fee_msat;
|
|
|
|
b->pay_msat += b->fee_msat;
|
|
|
|
a->fee_msat = b->fee_msat = 0;
|
|
|
|
|
|
|
|
/* BOLT #2:
|
|
|
|
*
|
|
|
|
* 1. If each nodes can afford half the fee from their
|
|
|
|
* to-`final_key` output, reduce the two to-`final_key`
|
|
|
|
* outputs accordingly.
|
|
|
|
*
|
|
|
|
* 2. Otherwise, reduce the to-`final_key` output of one node
|
|
|
|
* which cannot afford the fee to zero (resulting in that
|
|
|
|
* entire output paying fees). If the remaining
|
|
|
|
* to-`final_key` output is greater than the fee remaining,
|
|
|
|
* reduce it accordingly, otherwise reduce it to zero to
|
|
|
|
* pay as much fee as possible.
|
|
|
|
*/
|
|
|
|
remainder = pay_fee(a, fee_msat / 2) + pay_fee(b, fee_msat / 2);
|
|
|
|
|
|
|
|
/* If there's anything left, the other side tries to pay for it. */
|
|
|
|
remainder = pay_fee(a, remainder);
|
|
|
|
pay_fee(b, remainder);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* a transfers htlc_msat to a HTLC (gains it, if -ve) */
|
2016-03-15 07:37:30 +01:00
|
|
|
static bool change_funding(uint64_t anchor_satoshis,
|
2016-08-18 06:55:13 +02:00
|
|
|
uint64_t fee_rate,
|
2016-01-21 21:11:47 +01:00
|
|
|
int64_t htlc_msat,
|
2016-03-24 02:42:43 +01:00
|
|
|
struct channel_oneside *a,
|
|
|
|
struct channel_oneside *b,
|
2016-08-31 08:36:32 +02:00
|
|
|
size_t num_nondust_htlcs,
|
|
|
|
bool must_afford_fee)
|
2015-08-07 05:15:30 +02:00
|
|
|
{
|
2016-03-24 02:42:43 +01:00
|
|
|
uint64_t fee_msat;
|
2016-08-18 06:53:46 +02:00
|
|
|
uint64_t htlcs_total;
|
2015-08-07 05:15:30 +02:00
|
|
|
|
2016-08-18 06:53:46 +02:00
|
|
|
htlcs_total = anchor_satoshis * 1000
|
|
|
|
- (a->pay_msat + a->fee_msat + b->pay_msat + b->fee_msat);
|
2015-08-07 05:15:30 +02:00
|
|
|
|
2016-03-24 02:42:43 +01:00
|
|
|
fee_msat = calculate_fee_msat(num_nondust_htlcs, fee_rate);
|
2015-07-29 08:46:24 +02:00
|
|
|
|
2016-03-24 02:42:43 +01:00
|
|
|
/* If A is paying, can it afford it? */
|
2016-08-31 08:36:32 +02:00
|
|
|
if (htlc_msat >= 0) {
|
|
|
|
uint64_t cost = htlc_msat;
|
|
|
|
if (must_afford_fee)
|
|
|
|
cost += fee_msat / 2;
|
|
|
|
if (cost > a->pay_msat + a->fee_msat)
|
2016-03-24 02:42:43 +01:00
|
|
|
return false;
|
|
|
|
}
|
2015-08-07 05:15:30 +02:00
|
|
|
|
2016-03-24 02:42:43 +01:00
|
|
|
/* OK, now adjust funds for A, then recalculate fees. */
|
|
|
|
a->pay_msat -= htlc_msat;
|
|
|
|
recalculate_fees(a, b, fee_msat);
|
2015-08-07 05:15:30 +02:00
|
|
|
|
2016-08-18 06:53:46 +02:00
|
|
|
htlcs_total += htlc_msat;
|
|
|
|
assert(htlcs_total == anchor_satoshis * 1000
|
|
|
|
- (a->pay_msat + a->fee_msat + b->pay_msat + b->fee_msat));
|
2015-07-29 08:44:28 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2016-09-13 05:10:32 +02:00
|
|
|
bool anchor_too_large(uint64_t anchor_satoshis)
|
|
|
|
{
|
|
|
|
/* Anchor must fit in 32 bit. */
|
|
|
|
return anchor_satoshis >= (1ULL << 32) / 1000;
|
|
|
|
}
|
|
|
|
|
2016-06-30 01:38:10 +02:00
|
|
|
struct channel_state *initial_cstate(const tal_t *ctx,
|
2016-01-21 21:11:47 +01:00
|
|
|
uint64_t anchor_satoshis,
|
2016-08-18 06:55:13 +02:00
|
|
|
uint64_t fee_rate,
|
2016-08-31 08:36:32 +02:00
|
|
|
enum side funding)
|
2015-07-29 08:44:28 +02:00
|
|
|
{
|
2016-03-24 02:42:43 +01:00
|
|
|
uint64_t fee_msat;
|
2016-03-24 02:41:26 +01:00
|
|
|
struct channel_state *cstate = talz(ctx, struct channel_state);
|
2016-05-26 07:55:24 +02:00
|
|
|
struct channel_oneside *funder, *fundee;
|
2015-08-07 05:15:30 +02:00
|
|
|
|
2016-03-24 02:42:43 +01:00
|
|
|
cstate->fee_rate = fee_rate;
|
|
|
|
cstate->anchor = anchor_satoshis;
|
2016-08-18 06:53:46 +02:00
|
|
|
cstate->num_nondust = 0;
|
2016-03-24 02:42:43 +01:00
|
|
|
|
|
|
|
/* Anchor must fit in 32 bit. */
|
2016-09-13 05:10:32 +02:00
|
|
|
assert(!anchor_too_large(anchor_satoshis));
|
2015-08-07 05:15:30 +02:00
|
|
|
|
2016-03-24 02:42:43 +01:00
|
|
|
fee_msat = calculate_fee_msat(0, fee_rate);
|
|
|
|
if (fee_msat > anchor_satoshis * 1000)
|
2016-03-24 02:41:26 +01:00
|
|
|
return tal_free(cstate);
|
2016-03-24 02:42:43 +01:00
|
|
|
|
2016-05-26 07:55:24 +02:00
|
|
|
funder = &cstate->side[funding];
|
|
|
|
fundee = &cstate->side[!funding];
|
|
|
|
|
|
|
|
/* Neither side has HTLCs. */
|
2016-08-18 06:53:46 +02:00
|
|
|
funder->num_htlcs = fundee->num_htlcs = 0;
|
2015-08-07 05:15:30 +02:00
|
|
|
|
2016-05-26 07:55:24 +02:00
|
|
|
/* Initially, all goes back to funder. */
|
|
|
|
funder->pay_msat = anchor_satoshis * 1000 - fee_msat;
|
|
|
|
funder->fee_msat = fee_msat;
|
2015-07-29 08:44:28 +02:00
|
|
|
|
2016-01-21 21:11:47 +01:00
|
|
|
/* Make sure it checks out. */
|
2016-08-31 08:36:32 +02:00
|
|
|
assert(change_funding(anchor_satoshis, fee_rate, 0, funder, fundee, 0, false));
|
2016-05-26 07:55:24 +02:00
|
|
|
assert(funder->fee_msat == fee_msat);
|
|
|
|
assert(fundee->fee_msat == 0);
|
|
|
|
|
2016-03-24 02:41:26 +01:00
|
|
|
return cstate;
|
2015-07-29 08:44:28 +02:00
|
|
|
}
|
2015-07-29 08:46:24 +02:00
|
|
|
|
2016-08-23 02:42:00 +02:00
|
|
|
/* FIXME: Write exact variant! */
|
|
|
|
uint64_t approx_max_feerate(const struct channel_state *cstate,
|
2016-08-31 08:36:32 +02:00
|
|
|
enum side side)
|
2016-08-23 02:42:00 +02:00
|
|
|
{
|
|
|
|
uint64_t max_funds;
|
|
|
|
|
|
|
|
max_funds = cstate->side[side].pay_msat + cstate->side[side].fee_msat;
|
|
|
|
|
2016-10-07 05:30:17 +02:00
|
|
|
return max_funds / tx_bytes(cstate->num_nondust);
|
2016-08-23 02:42:00 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
bool can_afford_feerate(const struct channel_state *cstate, uint64_t fee_rate,
|
2016-08-31 08:36:32 +02:00
|
|
|
enum side side)
|
2016-08-23 02:42:00 +02:00
|
|
|
{
|
|
|
|
u64 fee_msat = calculate_fee_msat(cstate->num_nondust, fee_rate);
|
|
|
|
|
|
|
|
return cstate->side[side].pay_msat + cstate->side[side].fee_msat
|
|
|
|
>= fee_msat;
|
|
|
|
}
|
|
|
|
|
2016-08-18 06:55:13 +02:00
|
|
|
void adjust_fee(struct channel_state *cstate, uint64_t fee_rate)
|
2016-03-24 02:42:43 +01:00
|
|
|
{
|
|
|
|
uint64_t fee_msat;
|
|
|
|
|
2016-08-18 06:53:46 +02:00
|
|
|
fee_msat = calculate_fee_msat(cstate->num_nondust, fee_rate);
|
2016-03-24 02:42:43 +01:00
|
|
|
|
2016-08-31 08:36:32 +02:00
|
|
|
recalculate_fees(&cstate->side[LOCAL], &cstate->side[REMOTE], fee_msat);
|
2016-03-24 02:42:43 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
bool force_fee(struct channel_state *cstate, uint64_t fee)
|
|
|
|
{
|
|
|
|
/* Beware overflow! */
|
|
|
|
if (fee > 0xFFFFFFFFFFFFFFFFULL / 1000)
|
|
|
|
return false;
|
2016-08-31 08:36:32 +02:00
|
|
|
recalculate_fees(&cstate->side[LOCAL], &cstate->side[REMOTE], fee * 1000);
|
|
|
|
return cstate->side[LOCAL].fee_msat + cstate->side[REMOTE].fee_msat == fee * 1000;
|
2016-03-24 02:42:43 +01:00
|
|
|
}
|
2016-01-21 21:11:47 +01:00
|
|
|
|
2016-03-24 02:42:43 +01:00
|
|
|
/* Add a HTLC to @creator if it can afford it. */
|
2016-08-31 08:36:32 +02:00
|
|
|
bool cstate_add_htlc(struct channel_state *cstate, const struct htlc *htlc,
|
|
|
|
bool must_afford_fee)
|
2016-01-21 21:11:47 +01:00
|
|
|
{
|
2016-08-18 06:53:46 +02:00
|
|
|
size_t nondust;
|
2016-05-26 07:55:24 +02:00
|
|
|
struct channel_oneside *creator, *recipient;
|
2016-03-24 02:42:43 +01:00
|
|
|
|
2016-08-31 08:36:32 +02:00
|
|
|
creator = &cstate->side[htlc_owner(htlc)];
|
|
|
|
recipient = &cstate->side[!htlc_owner(htlc)];
|
2016-11-11 00:02:04 +01:00
|
|
|
|
2016-03-24 02:42:43 +01:00
|
|
|
/* Remember to count the new one in total txsize if not dust! */
|
2016-08-18 06:53:46 +02:00
|
|
|
nondust = cstate->num_nondust;
|
2016-09-06 09:17:49 +02:00
|
|
|
if (!is_dust(htlc->msatoshi / 1000))
|
2016-03-24 02:42:43 +01:00
|
|
|
nondust++;
|
2016-11-11 00:02:04 +01:00
|
|
|
|
2016-03-24 02:42:43 +01:00
|
|
|
if (!change_funding(cstate->anchor, cstate->fee_rate,
|
2016-09-06 09:17:49 +02:00
|
|
|
htlc->msatoshi, creator, recipient, nondust,
|
2016-08-31 08:36:32 +02:00
|
|
|
must_afford_fee))
|
2016-06-30 01:38:11 +02:00
|
|
|
return false;
|
2016-03-24 02:42:43 +01:00
|
|
|
|
2016-08-18 06:53:46 +02:00
|
|
|
cstate->num_nondust = nondust;
|
|
|
|
creator->num_htlcs++;
|
2016-06-30 01:38:11 +02:00
|
|
|
return true;
|
2016-03-24 02:42:43 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Remove htlc from creator, credit it to beneficiary. */
|
2016-03-31 08:42:20 +02:00
|
|
|
static void remove_htlc(struct channel_state *cstate,
|
2016-08-31 08:36:32 +02:00
|
|
|
enum side creator,
|
|
|
|
enum side beneficiary,
|
2016-08-18 06:53:46 +02:00
|
|
|
const struct htlc *htlc)
|
2016-03-24 02:42:43 +01:00
|
|
|
{
|
|
|
|
size_t nondust;
|
|
|
|
|
|
|
|
/* Remember to remove this one in total txsize if not dust! */
|
2016-08-18 06:53:46 +02:00
|
|
|
nondust = cstate->num_nondust;
|
2016-09-06 09:17:49 +02:00
|
|
|
if (!is_dust(htlc->msatoshi / 1000)) {
|
2016-03-24 02:42:43 +01:00
|
|
|
assert(nondust > 0);
|
|
|
|
nondust--;
|
|
|
|
}
|
|
|
|
|
2016-09-06 09:17:49 +02:00
|
|
|
/* Can't fail since msatoshi is positive. */
|
2016-03-24 02:42:43 +01:00
|
|
|
if (!change_funding(cstate->anchor, cstate->fee_rate,
|
2016-09-06 09:17:49 +02:00
|
|
|
-(int64_t)htlc->msatoshi,
|
2016-05-26 07:55:24 +02:00
|
|
|
&cstate->side[beneficiary],
|
2016-08-31 08:36:32 +02:00
|
|
|
&cstate->side[!beneficiary], nondust, false))
|
2016-03-24 02:42:43 +01:00
|
|
|
abort();
|
|
|
|
|
|
|
|
/* Actually remove the HTLC. */
|
2016-08-18 06:53:46 +02:00
|
|
|
assert(cstate->side[creator].num_htlcs > 0);
|
|
|
|
cstate->side[creator].num_htlcs--;
|
|
|
|
cstate->num_nondust = nondust;
|
2016-03-24 02:42:43 +01:00
|
|
|
}
|
|
|
|
|
2016-08-18 06:53:46 +02:00
|
|
|
void cstate_fail_htlc(struct channel_state *cstate, const struct htlc *htlc)
|
2016-03-24 02:42:43 +01:00
|
|
|
{
|
2016-08-31 08:36:32 +02:00
|
|
|
remove_htlc(cstate, htlc_owner(htlc), htlc_owner(htlc), htlc);
|
2016-03-24 02:42:43 +01:00
|
|
|
}
|
|
|
|
|
2016-08-18 06:53:46 +02:00
|
|
|
void cstate_fulfill_htlc(struct channel_state *cstate, const struct htlc *htlc)
|
2016-03-30 08:25:03 +02:00
|
|
|
{
|
2016-08-31 08:36:32 +02:00
|
|
|
remove_htlc(cstate, htlc_owner(htlc), !htlc_owner(htlc), htlc);
|
2016-03-30 08:25:03 +02:00
|
|
|
}
|
|
|
|
|
2016-06-30 01:38:10 +02:00
|
|
|
struct channel_state *copy_cstate(const tal_t *ctx,
|
2016-06-30 01:38:11 +02:00
|
|
|
const struct channel_state *cstate)
|
2016-01-21 21:11:47 +01:00
|
|
|
{
|
2016-08-18 06:53:46 +02:00
|
|
|
return tal_dup(ctx, struct channel_state, cstate);
|
2016-01-21 21:11:47 +01:00
|
|
|
}
|
2016-08-28 02:04:47 +02:00
|
|
|
|
|
|
|
void force_add_htlc(struct channel_state *cstate, const struct htlc *htlc)
|
|
|
|
{
|
|
|
|
struct channel_oneside *creator;
|
|
|
|
|
2016-08-31 08:36:32 +02:00
|
|
|
creator = &cstate->side[htlc_owner(htlc)];
|
2016-08-28 02:04:47 +02:00
|
|
|
creator->num_htlcs++;
|
2016-09-06 09:17:49 +02:00
|
|
|
creator->pay_msat -= htlc->msatoshi;
|
2016-11-11 00:02:04 +01:00
|
|
|
|
2016-08-28 02:04:47 +02:00
|
|
|
/* Remember to count the new one in total txsize if not dust! */
|
2016-09-06 09:17:49 +02:00
|
|
|
if (!is_dust(htlc->msatoshi / 1000))
|
2016-08-28 02:04:47 +02:00
|
|
|
cstate->num_nondust++;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void force_remove_htlc(struct channel_state *cstate,
|
2016-08-31 08:36:32 +02:00
|
|
|
enum side beneficiary,
|
2016-08-28 02:04:47 +02:00
|
|
|
const struct htlc *htlc)
|
|
|
|
{
|
2016-09-06 09:17:49 +02:00
|
|
|
cstate->side[beneficiary].pay_msat += htlc->msatoshi;
|
2016-08-31 08:36:32 +02:00
|
|
|
cstate->side[htlc_owner(htlc)].num_htlcs--;
|
2016-09-06 09:17:49 +02:00
|
|
|
if (!is_dust(htlc->msatoshi / 1000))
|
2016-08-28 02:04:47 +02:00
|
|
|
cstate->num_nondust--;
|
|
|
|
}
|
|
|
|
|
|
|
|
void force_fail_htlc(struct channel_state *cstate, const struct htlc *htlc)
|
|
|
|
{
|
2016-08-31 08:36:32 +02:00
|
|
|
force_remove_htlc(cstate, htlc_owner(htlc), htlc);
|
2016-08-28 02:04:47 +02:00
|
|
|
}
|
2016-11-11 00:02:04 +01:00
|
|
|
|
2016-08-28 02:04:47 +02:00
|
|
|
void force_fulfill_htlc(struct channel_state *cstate, const struct htlc *htlc)
|
|
|
|
{
|
2016-08-31 08:36:32 +02:00
|
|
|
force_remove_htlc(cstate, !htlc_owner(htlc), htlc);
|
2016-08-28 02:04:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
bool balance_after_force(struct channel_state *cstate)
|
|
|
|
{
|
|
|
|
/* We should not spend more than anchor */
|
2016-08-31 08:36:32 +02:00
|
|
|
if (cstate->side[LOCAL].pay_msat + cstate->side[REMOTE].pay_msat
|
2016-08-28 02:04:47 +02:00
|
|
|
> cstate->anchor * 1000)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
/* Check for wrap. */
|
2016-08-31 08:36:32 +02:00
|
|
|
if (cstate->side[LOCAL].pay_msat > cstate->anchor * 1000)
|
2016-08-28 02:04:47 +02:00
|
|
|
return false;
|
2016-08-31 08:36:32 +02:00
|
|
|
if (cstate->side[REMOTE].pay_msat > cstate->anchor * 1000)
|
2016-08-28 02:04:47 +02:00
|
|
|
return false;
|
|
|
|
|
|
|
|
if (cstate->num_nondust
|
2016-08-31 08:36:32 +02:00
|
|
|
> cstate->side[LOCAL].num_htlcs + cstate->side[REMOTE].num_htlcs)
|
2016-08-28 02:04:47 +02:00
|
|
|
return false;
|
|
|
|
|
|
|
|
/* Recalc fees. */
|
|
|
|
adjust_fee(cstate, cstate->fee_rate);
|
|
|
|
return true;
|
|
|
|
}
|
2017-01-04 04:37:15 +01:00
|
|
|
|
|
|
|
static char *fmt_channel_oneside(const tal_t *ctx,
|
|
|
|
const struct channel_oneside *co)
|
|
|
|
{
|
|
|
|
return tal_fmt(ctx, "{ pay_msat=%u"
|
|
|
|
" fee_msat=%u"
|
|
|
|
" num_htlcs=%u }",
|
|
|
|
co->pay_msat,
|
|
|
|
co->fee_msat,
|
|
|
|
co->num_htlcs);
|
|
|
|
}
|
|
|
|
|
|
|
|
static char *fmt_channel_state(const tal_t *ctx,
|
|
|
|
const struct channel_state *cs)
|
|
|
|
{
|
|
|
|
return tal_fmt(ctx, "{ anchor=%"PRIu64
|
|
|
|
" fee_rate=%"PRIu64
|
|
|
|
" num_nondust=%u"
|
|
|
|
" ours=%s"
|
|
|
|
" theirs=%s }",
|
|
|
|
cs->anchor,
|
|
|
|
cs->fee_rate,
|
|
|
|
cs->num_nondust,
|
|
|
|
fmt_channel_oneside(ctx, &cs->side[LOCAL]),
|
|
|
|
fmt_channel_oneside(ctx, &cs->side[REMOTE]));
|
|
|
|
}
|
|
|
|
|
|
|
|
REGISTER_TYPE_TO_STRING(channel_oneside, fmt_channel_oneside);
|
|
|
|
REGISTER_TYPE_TO_STRING(channel_state, fmt_channel_state);
|