#include "config.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include /* stdin == requests, 3 == peer, 4 = gossip, 5 = hsmd */ #define REQ_FD STDIN_FILENO #define HSM_FD 5 static void notify(enum log_level level, const char *fmt, ...) { va_list ap; va_start(ap, fmt); wire_sync_write(REQ_FD, take(towire_closingd_notification(NULL, level, tal_vfmt(tmpctx, fmt, ap)))); va_end(ap); } static struct bitcoin_tx *close_tx(const tal_t *ctx, const struct chainparams *chainparams, struct per_peer_state *pps, const struct channel_id *channel_id, u8 *scriptpubkey[NUM_SIDES], const struct bitcoin_outpoint *funding, struct amount_sat funding_sats, const u8 *funding_wscript, const struct amount_sat out[NUM_SIDES], enum side opener, struct amount_sat fee, struct amount_sat dust_limit, const struct bitcoin_outpoint *wrong_funding) { struct bitcoin_tx *tx; struct amount_sat out_minus_fee[NUM_SIDES]; out_minus_fee[LOCAL] = out[LOCAL]; out_minus_fee[REMOTE] = out[REMOTE]; if (!amount_sat_sub(&out_minus_fee[opener], out[opener], fee)) peer_failed_warn(pps, channel_id, "Funder cannot afford fee %s (%s and %s)", type_to_string(tmpctx, struct amount_sat, &fee), type_to_string(tmpctx, struct amount_sat, &out[LOCAL]), type_to_string(tmpctx, struct amount_sat, &out[REMOTE])); status_debug("Making close tx at = %s/%s fee %s", type_to_string(tmpctx, struct amount_sat, &out[LOCAL]), type_to_string(tmpctx, struct amount_sat, &out[REMOTE]), type_to_string(tmpctx, struct amount_sat, &fee)); /* FIXME: We need to allow this! */ tx = create_close_tx(ctx, chainparams, scriptpubkey[LOCAL], scriptpubkey[REMOTE], funding_wscript, funding, funding_sats, out_minus_fee[LOCAL], out_minus_fee[REMOTE], dust_limit); if (!tx) peer_failed_err(pps, channel_id, "Both outputs below dust limit:" " funding = %s" " fee = %s" " dust_limit = %s" " LOCAL = %s" " REMOTE = %s", type_to_string(tmpctx, struct amount_sat, &funding_sats), type_to_string(tmpctx, struct amount_sat, &fee), type_to_string(tmpctx, struct amount_sat, &dust_limit), type_to_string(tmpctx, struct amount_sat, &out[LOCAL]), type_to_string(tmpctx, struct amount_sat, &out[REMOTE])); if (wrong_funding) bitcoin_tx_input_set_outpoint(tx, 0, wrong_funding); return tx; } /* Handle random messages we might get, returning the first non-handled one. */ static u8 *closing_read_peer_msg(const tal_t *ctx, struct per_peer_state *pps, const struct channel_id *channel_id) { for (;;) { u8 *msg; bool from_gossipd; clean_tmpctx(); msg = peer_or_gossip_sync_read(ctx, pps, &from_gossipd); if (from_gossipd) { handle_gossip_msg(pps, take(msg)); continue; } /* Handle custommsgs */ enum peer_wire type = fromwire_peektype(msg); if (type % 2 == 1 && !peer_wire_is_defined(type)) { /* The message is not part of the messages we know * how to handle. Assume is custommsg, forward it * to master. */ wire_sync_write(REQ_FD, take(towire_custommsg_in(NULL, msg))); continue; } if (!handle_peer_gossip_or_error(pps, channel_id, false, msg)) return msg; } } static void send_offer(struct per_peer_state *pps, const struct chainparams *chainparams, const struct channel_id *channel_id, const struct pubkey funding_pubkey[NUM_SIDES], const u8 *funding_wscript, u8 *scriptpubkey[NUM_SIDES], const struct bitcoin_outpoint *funding, struct amount_sat funding_sats, const struct amount_sat out[NUM_SIDES], enum side opener, struct amount_sat our_dust_limit, struct amount_sat fee_to_offer, const struct bitcoin_outpoint *wrong_funding, const struct tlv_closing_signed_tlvs_fee_range *tlv_fees) { struct bitcoin_tx *tx; struct bitcoin_signature our_sig; struct tlv_closing_signed_tlvs *close_tlvs; u8 *msg; /* BOLT #2: * * - MUST set `signature` to the Bitcoin signature of the close * transaction, as specified in [BOLT * #3](03-transactions.md#closing-transaction). */ tx = close_tx(tmpctx, chainparams, pps, channel_id, scriptpubkey, funding, funding_sats, funding_wscript, out, opener, fee_to_offer, our_dust_limit, wrong_funding); /* BOLT #3: * * ## Closing Transaction *... * Each node offering a signature... MAY eliminate its * own output. */ /* (We don't do this). */ wire_sync_write(HSM_FD, take(towire_hsmd_sign_mutual_close_tx(NULL, tx, &funding_pubkey[REMOTE]))); msg = wire_sync_read(tmpctx, HSM_FD); if (!fromwire_hsmd_sign_tx_reply(msg, &our_sig)) status_failed(STATUS_FAIL_HSM_IO, "Bad hsm_sign_mutual_close_tx reply %s", tal_hex(tmpctx, msg)); status_debug("sending fee offer %s", type_to_string(tmpctx, struct amount_sat, &fee_to_offer)); /* Add the new close_tlvs with our fee range */ if (tlv_fees) { close_tlvs = tlv_closing_signed_tlvs_new(msg); close_tlvs->fee_range = cast_const(struct tlv_closing_signed_tlvs_fee_range *, tlv_fees); notify(LOG_INFORM, "Sending closing fee offer %s, with range %s-%s", type_to_string(tmpctx, struct amount_sat, &fee_to_offer), type_to_string(tmpctx, struct amount_sat, &tlv_fees->min_fee_satoshis), type_to_string(tmpctx, struct amount_sat, &tlv_fees->max_fee_satoshis)); } else close_tlvs = NULL; assert(our_sig.sighash_type == SIGHASH_ALL); msg = towire_closing_signed(NULL, channel_id, fee_to_offer, &our_sig.s, close_tlvs); peer_write(pps, take(msg)); } static void tell_master_their_offer(const struct bitcoin_signature *their_sig, const struct bitcoin_tx *tx, struct bitcoin_txid *tx_id) { u8 *msg = towire_closingd_received_signature(NULL, their_sig, tx); if (!wire_sync_write(REQ_FD, take(msg))) status_failed(STATUS_FAIL_MASTER_IO, "Writing received to master: %s", strerror(errno)); /* Wait for master to ack, to make sure it's in db. */ msg = wire_sync_read(NULL, REQ_FD); if (!fromwire_closingd_received_signature_reply(msg, tx_id)) master_badmsg(WIRE_CLOSINGD_RECEIVED_SIGNATURE_REPLY, msg); tal_free(msg); } /* Returns fee they offered. */ static struct amount_sat receive_offer(struct per_peer_state *pps, const struct chainparams *chainparams, const struct channel_id *channel_id, const struct pubkey funding_pubkey[NUM_SIDES], const u8 *funding_wscript, u8 *scriptpubkey[NUM_SIDES], const struct bitcoin_outpoint *funding, struct amount_sat funding_sats, const struct amount_sat out[NUM_SIDES], enum side opener, struct amount_sat our_dust_limit, struct amount_sat min_fee_to_accept, const struct bitcoin_outpoint *wrong_funding, struct bitcoin_txid *closing_txid, struct tlv_closing_signed_tlvs_fee_range **tlv_fees) { u8 *msg; struct channel_id their_channel_id; struct amount_sat received_fee; struct bitcoin_signature their_sig; struct bitcoin_tx *tx; struct tlv_closing_signed_tlvs *close_tlvs; /* Wait for them to say something interesting */ do { msg = closing_read_peer_msg(tmpctx, pps, channel_id); /* BOLT #2: * * - upon reconnection: * - MUST ignore any redundant `funding_locked` it receives. */ /* This should only happen if we've made no commitments, but * we don't have to check that: it's their problem. */ if (fromwire_peektype(msg) == WIRE_FUNDING_LOCKED) msg = tal_free(msg); /* BOLT #2: * - if it has sent a previous `shutdown`: * - MUST retransmit `shutdown`. */ else if (fromwire_peektype(msg) == WIRE_SHUTDOWN) msg = tal_free(msg); /* channeld may have sent ping: ignore pong! */ else if (fromwire_peektype(msg) == WIRE_PONG) msg = tal_free(msg); } while (!msg); their_sig.sighash_type = SIGHASH_ALL; close_tlvs = tlv_closing_signed_tlvs_new(msg); if (!fromwire_closing_signed(msg, &their_channel_id, &received_fee, &their_sig.s, close_tlvs)) peer_failed_warn(pps, channel_id, "Expected closing_signed: %s", tal_hex(tmpctx, msg)); /* BOLT #2: * * The receiving node: * - if the `signature` is not valid for either variant of closing transaction * specified in [BOLT #3](03-transactions.md#closing-transaction) * OR non-compliant with LOW-S-standard rule...: * - MUST fail the connection. */ tx = close_tx(tmpctx, chainparams, pps, channel_id, scriptpubkey, funding, funding_sats, funding_wscript, out, opener, received_fee, our_dust_limit, wrong_funding); if (!check_tx_sig(tx, 0, NULL, funding_wscript, &funding_pubkey[REMOTE], &their_sig)) { /* Trim it by reducing their output to minimum */ struct bitcoin_tx *trimmed; struct amount_sat trimming_out[NUM_SIDES]; if (opener == REMOTE) trimming_out[REMOTE] = received_fee; else trimming_out[REMOTE] = AMOUNT_SAT(0); trimming_out[LOCAL] = out[LOCAL]; /* BOLT #3: * * Each node offering a signature: * - MUST round each output down to whole satoshis. * - MUST subtract the fee given by `fee_satoshis` from the * output to the funder. * - MUST remove any output below its own * `dust_limit_satoshis`. * - MAY eliminate its own output. */ trimmed = close_tx(tmpctx, chainparams, pps, channel_id, scriptpubkey, funding, funding_sats, funding_wscript, trimming_out, opener, received_fee, our_dust_limit, wrong_funding); if (!trimmed || !check_tx_sig(trimmed, 0, NULL, funding_wscript, &funding_pubkey[REMOTE], &their_sig)) { peer_failed_warn(pps, channel_id, "Bad closing_signed signature for" " %s (and trimmed version %s)", type_to_string(tmpctx, struct bitcoin_tx, tx), trimmed ? type_to_string(tmpctx, struct bitcoin_tx, trimmed) : "NONE"); } tx = trimmed; } status_debug("Received fee offer %s", type_to_string(tmpctx, struct amount_sat, &received_fee)); if (tlv_fees) { if (close_tlvs) { *tlv_fees = tal_steal(tlv_fees, close_tlvs->fee_range); } else { *tlv_fees = NULL; } } if (close_tlvs && close_tlvs->fee_range) { notify(LOG_INFORM, "Received closing fee offer %s, with range %s-%s", type_to_string(tmpctx, struct amount_sat, &received_fee), type_to_string(tmpctx, struct amount_sat, &close_tlvs->fee_range->min_fee_satoshis), type_to_string(tmpctx, struct amount_sat, &close_tlvs->fee_range->max_fee_satoshis)); } else { notify(LOG_INFORM, "Received closing fee offer %s, without range", type_to_string(tmpctx, struct amount_sat, &received_fee)); } /* Master sorts out what is best offer, we just tell it any above min */ if (amount_sat_greater_eq(received_fee, min_fee_to_accept)) { status_debug("...offer is reasonable"); tell_master_their_offer(&their_sig, tx, closing_txid); } return received_fee; } struct feerange { enum side higher_side; struct amount_sat min, max; }; static void init_feerange(struct feerange *feerange, struct amount_sat commitment_fee, const struct amount_sat offer[NUM_SIDES]) { feerange->min = AMOUNT_SAT(0); /* FIXME: BOLT 2 previously said that we have to set it to less than * the final commit fee: we do this for now, still: * * - MUST set `fee_satoshis` less than or equal to the base * fee of the final commitment transaction, as calculated * in [BOLT #3](03-transactions.md#fee-calculation). */ feerange->max = commitment_fee; if (amount_sat_greater(offer[LOCAL], offer[REMOTE])) feerange->higher_side = LOCAL; else feerange->higher_side = REMOTE; status_debug("Feerange init %s-%s, %s higher", type_to_string(tmpctx, struct amount_sat, &feerange->min), type_to_string(tmpctx, struct amount_sat, &feerange->max), feerange->higher_side == LOCAL ? "local" : "remote"); } static void adjust_feerange(struct feerange *feerange, struct amount_sat offer, enum side side) { bool ok; /* FIXME: BOLT 2 previously said that we have to set it to less than * the final commit fee: we do this for now, still: * * - MUST propose a value "strictly between" the received * `fee_satoshis` and its previously-sent `fee_satoshis`. */ if (side == feerange->higher_side) ok = amount_sat_sub(&feerange->max, offer, AMOUNT_SAT(1)); else ok = amount_sat_add(&feerange->min, offer, AMOUNT_SAT(1)); status_debug("Feerange %s update %s: now %s-%s", side == LOCAL ? "local" : "remote", type_to_string(tmpctx, struct amount_sat, &offer), type_to_string(tmpctx, struct amount_sat, &feerange->min), type_to_string(tmpctx, struct amount_sat, &feerange->max)); if (!ok) status_failed(STATUS_FAIL_INTERNAL_ERROR, "Overflow in updating fee range"); } /* Do these two ranges overlap? If so, return that range. */ static bool get_overlap(const struct tlv_closing_signed_tlvs_fee_range *r1, const struct tlv_closing_signed_tlvs_fee_range *r2, struct tlv_closing_signed_tlvs_fee_range *overlap) { if (amount_sat_greater(r1->min_fee_satoshis, r2->min_fee_satoshis)) overlap->min_fee_satoshis = r1->min_fee_satoshis; else overlap->min_fee_satoshis = r2->min_fee_satoshis; if (amount_sat_less(r1->max_fee_satoshis, r2->max_fee_satoshis)) overlap->max_fee_satoshis = r1->max_fee_satoshis; else overlap->max_fee_satoshis = r2->max_fee_satoshis; return amount_sat_less_eq(overlap->min_fee_satoshis, overlap->max_fee_satoshis); } /* Is this amount in this range? */ static bool amount_in_range(struct amount_sat amount, const struct tlv_closing_signed_tlvs_fee_range *r) { return amount_sat_greater_eq(amount, r->min_fee_satoshis) && amount_sat_less_eq(amount, r->max_fee_satoshis); } /* Figure out what we should offer now. */ static struct amount_sat adjust_offer(struct per_peer_state *pps, const struct channel_id *channel_id, const struct feerange *feerange, struct amount_sat remote_offer, struct amount_sat min_fee_to_accept, u64 fee_negotiation_step, u8 fee_negotiation_step_unit) { struct amount_sat min_plus_one, range_len, step_sat, result; struct amount_msat step_msat; /* Within 1 satoshi? Agree. */ if (!amount_sat_add(&min_plus_one, feerange->min, AMOUNT_SAT(1))) peer_failed_warn(pps, channel_id, "Fee offer %s min too large", type_to_string(tmpctx, struct amount_sat, &feerange->min)); if (amount_sat_greater_eq(min_plus_one, feerange->max)) return remote_offer; /* feerange has already been adjusted so that our new offer is ok to be * any number in [feerange->min, feerange->max] and after the following * min_fee_to_accept is in that range. Thus, pick a fee in * [min_fee_to_accept, feerange->max]. */ if (amount_sat_greater(feerange->min, min_fee_to_accept)) min_fee_to_accept = feerange->min; /* Max is below our minimum acceptable? */ if (!amount_sat_sub(&range_len, feerange->max, min_fee_to_accept)) peer_failed_warn(pps, channel_id, "Feerange %s-%s" " below minimum acceptable %s", type_to_string(tmpctx, struct amount_sat, &feerange->min), type_to_string(tmpctx, struct amount_sat, &feerange->max), type_to_string(tmpctx, struct amount_sat, &min_fee_to_accept)); if (fee_negotiation_step_unit == CLOSING_FEE_NEGOTIATION_STEP_UNIT_SATOSHI) { /* -1 because the range boundary has already been adjusted with * one from our previous proposal. So, if the user requested a * step of 1 satoshi at a time we should just return our end of * the range from this function. */ step_msat = amount_msat((fee_negotiation_step - 1) * MSAT_PER_SAT); } else { /* fee_negotiation_step is e.g. 20 to designate 20% from * range_len (which is in satoshi), so: * range_len * fee_negotiation_step / 100 [sat] * is equivalent to: * range_len * fee_negotiation_step * 10 [msat] */ step_msat = amount_msat(range_len.satoshis /* Raw: % calc */ * fee_negotiation_step * 10); } step_sat = amount_msat_to_sat_round_down(step_msat); if (feerange->higher_side == LOCAL) { if (!amount_sat_sub(&result, feerange->max, step_sat)) /* step_sat > feerange->max, unlikely */ return min_fee_to_accept; if (amount_sat_less_eq(result, min_fee_to_accept)) return min_fee_to_accept; } else { if (!amount_sat_add(&result, min_fee_to_accept, step_sat)) /* overflow, unlikely */ return feerange->max; if (amount_sat_greater_eq(result, feerange->max)) return feerange->max; } return result; } #if DEVELOPER /* FIXME: We should talk to lightningd anyway, rather than doing this */ static void closing_dev_memleak(const tal_t *ctx, u8 *scriptpubkey[NUM_SIDES], const u8 *funding_wscript) { struct htable *memtable; memtable = memleak_find_allocations(tmpctx, NULL, NULL); memleak_remove_pointer(memtable, ctx); memleak_remove_pointer(memtable, scriptpubkey[LOCAL]); memleak_remove_pointer(memtable, scriptpubkey[REMOTE]); memleak_remove_pointer(memtable, funding_wscript); dump_memleak(memtable, memleak_status_broken); } #endif /* DEVELOPER */ /* Figure out what weight we actually expect for this closing tx (using zero fees * gives the largest possible tx: larger values might omit outputs). */ static size_t closing_tx_weight_estimate(u8 *scriptpubkey[NUM_SIDES], const u8 *funding_wscript, const struct amount_sat *out, struct amount_sat funding_sats, struct amount_sat dust_limit) { /* We create a dummy close */ struct bitcoin_tx *tx; struct bitcoin_outpoint dummy_funding; struct bitcoin_signature dummy_sig; struct privkey dummy_privkey; struct pubkey dummy_pubkey; u8 **witness; memset(&dummy_funding, 0, sizeof(dummy_funding)); tx = create_close_tx(tmpctx, chainparams, scriptpubkey[LOCAL], scriptpubkey[REMOTE], funding_wscript, &dummy_funding, funding_sats, out[LOCAL], out[REMOTE], dust_limit); /* Create a signature, any signature, so we can weigh fully "signed" * tx. */ dummy_sig.sighash_type = SIGHASH_ALL; memset(&dummy_privkey, 1, sizeof(dummy_privkey)); sign_hash(&dummy_privkey, &dummy_funding.txid.shad, &dummy_sig.s); pubkey_from_privkey(&dummy_privkey, &dummy_pubkey); witness = bitcoin_witness_2of2(NULL, &dummy_sig, &dummy_sig, &dummy_pubkey, &dummy_pubkey); bitcoin_tx_input_set_witness(tx, 0, take(witness)); return bitcoin_tx_weight(tx); } /* Get the minimum and desired fees */ static void calc_fee_bounds(size_t expected_weight, u32 min_feerate, u32 desired_feerate, u32 *max_feerate, struct amount_sat commitment_fee, struct amount_sat funding, enum side opener, struct amount_sat *minfee, struct amount_sat *desiredfee, struct amount_sat *maxfee) { *minfee = amount_tx_fee(min_feerate, expected_weight); *desiredfee = amount_tx_fee(desired_feerate, expected_weight); /* BOLT #2: * - if it is not the funder: * - SHOULD set `max_fee_satoshis` to at least the `max_fee_satoshis` * received *... * Note that the non-funder is not paying the fee, so there is * no reason for it to have a maximum feerate. */ if (opener == REMOTE) { *maxfee = funding; /* This used to appear in BOLT #2: we still set it for non-anchor * peers who may still enforce it: * - If the channel does not use `option_anchor_outputs`: * - MUST set `fee_satoshis` less than or equal to the base fee of * the final commitment transaction, as calculated in * [BOLT #3](03-transactions.md#fee-calculation). */ } else if (max_feerate) { *maxfee = amount_tx_fee(*max_feerate, expected_weight); status_debug("deriving max fee from rate %u -> %s (not %s)", *max_feerate, type_to_string(tmpctx, struct amount_sat, maxfee), type_to_string(tmpctx, struct amount_sat, &commitment_fee)); /* option_anchor_outputs sets commitment_fee to max, so this * doesn't do anything */ if (amount_sat_greater(*maxfee, commitment_fee)) { /* FIXME: would be nice to notify close cmd here! */ status_unusual("Maximum feerate %u would give fee %s:" " we must limit it to the final commitment fee %s", *max_feerate, type_to_string(tmpctx, struct amount_sat, maxfee), type_to_string(tmpctx, struct amount_sat, &commitment_fee)); *maxfee = commitment_fee; } } else *maxfee = commitment_fee; /* Can't exceed maxfee. */ if (amount_sat_greater(*minfee, *maxfee)) *minfee = *maxfee; if (amount_sat_less(*desiredfee, *minfee)) { status_unusual("Our ideal fee is %s (%u sats/perkw)," " but our minimum is %s: using that", type_to_string(tmpctx, struct amount_sat, desiredfee), desired_feerate, type_to_string(tmpctx, struct amount_sat, minfee)); *desiredfee = *minfee; } if (amount_sat_greater(*desiredfee, *maxfee)) { status_unusual("Our ideal fee is %s (%u sats/perkw)," " but our maximum is %s: using that", type_to_string(tmpctx, struct amount_sat, desiredfee), desired_feerate, type_to_string(tmpctx, struct amount_sat, maxfee)); *desiredfee = *maxfee; } status_debug("Expected closing weight = %zu, fee %s (min %s, max %s)", expected_weight, type_to_string(tmpctx, struct amount_sat, desiredfee), type_to_string(tmpctx, struct amount_sat, minfee), type_to_string(tmpctx, struct amount_sat, maxfee)); } /* We've received one offer; if we're opener, that means we've already sent one * too. */ static void do_quickclose(struct amount_sat offer[NUM_SIDES], struct per_peer_state *pps, const struct channel_id *channel_id, const struct pubkey funding_pubkey[NUM_SIDES], const u8 *funding_wscript, u8 *scriptpubkey[NUM_SIDES], const struct bitcoin_outpoint *funding, struct amount_sat funding_sats, const struct amount_sat out[NUM_SIDES], enum side opener, struct amount_sat our_dust_limit, const struct bitcoin_outpoint *wrong_funding, struct bitcoin_txid *closing_txid, const struct tlv_closing_signed_tlvs_fee_range *our_feerange, const struct tlv_closing_signed_tlvs_fee_range *their_feerange) { struct tlv_closing_signed_tlvs_fee_range overlap; /* BOLT #2: * - if the message contains a `fee_range`: * - if there is no overlap between that and its own `fee_range`: * - SHOULD fail the connection */ if (!get_overlap(our_feerange, their_feerange, &overlap)) { peer_failed_warn(pps, channel_id, "Unable to agree on a feerate." " Our range %s-%s, other range %s-%s", type_to_string(tmpctx, struct amount_sat, &our_feerange->min_fee_satoshis), type_to_string(tmpctx, struct amount_sat, &our_feerange->max_fee_satoshis), type_to_string(tmpctx, struct amount_sat, &their_feerange->min_fee_satoshis), type_to_string(tmpctx, struct amount_sat, &their_feerange->max_fee_satoshis)); return; } status_info("performing quickclose in range %s-%s", type_to_string(tmpctx, struct amount_sat, &overlap.min_fee_satoshis), type_to_string(tmpctx, struct amount_sat, &overlap.max_fee_satoshis)); /* BOLT #2: * - otherwise: * - if it is the funder: * - if `fee_satoshis` is not in the overlap between the sent * and received `fee_range`: * - MUST fail the channel * - otherwise: * - MUST reply with the same `fee_satoshis`. */ if (opener == LOCAL) { if (!amount_in_range(offer[REMOTE], &overlap)) { peer_failed_warn(pps, channel_id, "Your fee %s was not in range:" " Our range %s-%s, other range %s-%s", type_to_string(tmpctx, struct amount_sat, &offer[REMOTE]), type_to_string(tmpctx, struct amount_sat, &our_feerange->min_fee_satoshis), type_to_string(tmpctx, struct amount_sat, &our_feerange->max_fee_satoshis), type_to_string(tmpctx, struct amount_sat, &their_feerange->min_fee_satoshis), type_to_string(tmpctx, struct amount_sat, &their_feerange->max_fee_satoshis)); return; } /* Only reply if we didn't already completely agree. */ if (!amount_sat_eq(offer[LOCAL], offer[REMOTE])) { offer[LOCAL] = offer[REMOTE]; send_offer(pps, chainparams, channel_id, funding_pubkey, funding_wscript, scriptpubkey, funding, funding_sats, out, opener, our_dust_limit, offer[LOCAL], wrong_funding, our_feerange); } } else { /* BOLT #2: * - otherwise (it is not the funder): * - if it has already sent a `closing_signed`: * - if `fee_satoshis` is not the same as the value it sent: * - MUST fail the channel * - otherwise: * - MUST propose a `fee_satoshis` in the overlap between * received and (about-to-be) sent `fee_range`. */ if (!amount_in_range(offer[LOCAL], &overlap)) { /* Hmm, go to edges. */ if (amount_sat_greater(offer[LOCAL], overlap.max_fee_satoshis)) { offer[LOCAL] = overlap.max_fee_satoshis; status_unusual("Lowered offer to max allowable" " %s", type_to_string(tmpctx, struct amount_sat, &offer[LOCAL])); } else if (amount_sat_less(offer[LOCAL], overlap.min_fee_satoshis)) { offer[LOCAL] = overlap.min_fee_satoshis; status_unusual("Increased offer to min allowable" " %s", type_to_string(tmpctx, struct amount_sat, &offer[LOCAL])); } } send_offer(pps, chainparams, channel_id, funding_pubkey, funding_wscript, scriptpubkey, funding, funding_sats, out, opener, our_dust_limit, offer[LOCAL], wrong_funding, our_feerange); /* They will reply unless we completely agreed. */ if (!amount_sat_eq(offer[LOCAL], offer[REMOTE])) { offer[REMOTE] = receive_offer(pps, chainparams, channel_id, funding_pubkey, funding_wscript, scriptpubkey, funding, funding_sats, out, opener, our_dust_limit, our_feerange->min_fee_satoshis, wrong_funding, closing_txid, NULL); /* BOLT #2: * - otherwise (it is not the funder): * - if it has already sent a `closing_signed`: * - if `fee_satoshis` is not the same as the value * it sent: * - MUST fail the channel */ if (!amount_sat_eq(offer[LOCAL], offer[REMOTE])) { peer_failed_warn(pps, channel_id, "Your fee %s was not equal to %s", type_to_string(tmpctx, struct amount_sat, &offer[REMOTE]), type_to_string(tmpctx, struct amount_sat, &offer[LOCAL])); return; } } } peer_billboard(true, "We agreed on a closing fee of %"PRIu64" satoshi for tx:%s", offer[LOCAL], type_to_string(tmpctx, struct bitcoin_txid, closing_txid)); } int main(int argc, char *argv[]) { setup_locale(); const tal_t *ctx = tal(NULL, char); struct per_peer_state *pps; u8 *msg; struct pubkey funding_pubkey[NUM_SIDES]; struct bitcoin_txid closing_txid; struct bitcoin_outpoint funding; struct amount_sat funding_sats, out[NUM_SIDES]; struct amount_sat our_dust_limit; struct amount_sat min_fee_to_accept, commitment_fee, offer[NUM_SIDES], max_fee_to_accept; u32 min_feerate, initial_feerate, *max_feerate; struct feerange feerange; enum side opener; u8 *scriptpubkey[NUM_SIDES], *funding_wscript; u64 fee_negotiation_step; u8 fee_negotiation_step_unit; char fee_negotiation_step_str[32]; /* fee_negotiation_step + "sat" */ struct channel_id channel_id; enum side whose_turn; bool use_quickclose; struct tlv_closing_signed_tlvs_fee_range *our_feerange, **their_feerange; struct bitcoin_outpoint *wrong_funding; subdaemon_setup(argc, argv); status_setup_sync(REQ_FD); msg = wire_sync_read(tmpctx, REQ_FD); if (!fromwire_closingd_init(ctx, msg, &chainparams, &channel_id, &funding, &funding_sats, &funding_pubkey[LOCAL], &funding_pubkey[REMOTE], &opener, &out[LOCAL], &out[REMOTE], &our_dust_limit, &min_feerate, &initial_feerate, &max_feerate, &commitment_fee, &scriptpubkey[LOCAL], &scriptpubkey[REMOTE], &fee_negotiation_step, &fee_negotiation_step_unit, &use_quickclose, &wrong_funding)) master_badmsg(WIRE_CLOSINGD_INIT, msg); /* stdin == requests, 3 == peer, 4 = gossip, 5 = hsmd */ pps = notleak(new_per_peer_state(ctx)); per_peer_state_set_fds(pps, 3, 4); funding_wscript = bitcoin_redeem_2of2(ctx, &funding_pubkey[LOCAL], &funding_pubkey[REMOTE]); /* Start at what we consider a reasonable feerate for this tx. */ calc_fee_bounds(closing_tx_weight_estimate(scriptpubkey, funding_wscript, out, funding_sats, our_dust_limit), min_feerate, initial_feerate, max_feerate, commitment_fee, funding_sats, opener, &min_fee_to_accept, &offer[LOCAL], &max_fee_to_accept); /* Write values into tlv for updated closing fee neg */ their_feerange = tal(ctx, struct tlv_closing_signed_tlvs_fee_range *); *their_feerange = NULL; if (use_quickclose) { our_feerange = tal(ctx, struct tlv_closing_signed_tlvs_fee_range); our_feerange->min_fee_satoshis = min_fee_to_accept; our_feerange->max_fee_satoshis = max_fee_to_accept; } else our_feerange = NULL; snprintf(fee_negotiation_step_str, sizeof(fee_negotiation_step_str), "%" PRIu64 "%s", fee_negotiation_step, fee_negotiation_step_unit == CLOSING_FEE_NEGOTIATION_STEP_UNIT_PERCENTAGE ? "%" : "sat"); status_debug("out = %s/%s", type_to_string(tmpctx, struct amount_sat, &out[LOCAL]), type_to_string(tmpctx, struct amount_sat, &out[REMOTE])); status_debug("dustlimit = %s", type_to_string(tmpctx, struct amount_sat, &our_dust_limit)); status_debug("fee = %s", type_to_string(tmpctx, struct amount_sat, &offer[LOCAL])); status_debug("fee negotiation step = %s", fee_negotiation_step_str); if (wrong_funding) status_unusual("Setting wrong_funding_txid to %s:%u", type_to_string(tmpctx, struct bitcoin_txid, &wrong_funding->txid), wrong_funding->n); peer_billboard( true, "Negotiating closing fee between %s and %s satoshi (ideal %s) " "using step %s", type_to_string(tmpctx, struct amount_sat, &min_fee_to_accept), type_to_string(tmpctx, struct amount_sat, &max_fee_to_accept), type_to_string(tmpctx, struct amount_sat, &offer[LOCAL]), fee_negotiation_step_str); /* BOLT #2: * * The funding node: * - after `shutdown` has been received, AND no HTLCs remain in either * commitment transaction: * - SHOULD send a `closing_signed` message. */ whose_turn = opener; for (size_t i = 0; i < 2; i++, whose_turn = !whose_turn) { if (whose_turn == LOCAL) { send_offer(pps, chainparams, &channel_id, funding_pubkey, funding_wscript, scriptpubkey, &funding, funding_sats, out, opener, our_dust_limit, offer[LOCAL], wrong_funding, our_feerange); } else { if (i == 0) peer_billboard(false, "Waiting for their initial" " closing fee offer"); else peer_billboard(false, "Waiting for their initial" " closing fee offer:" " ours was %s", type_to_string(tmpctx, struct amount_sat, &offer[LOCAL])); offer[REMOTE] = receive_offer(pps, chainparams, &channel_id, funding_pubkey, funding_wscript, scriptpubkey, &funding, funding_sats, out, opener, our_dust_limit, min_fee_to_accept, wrong_funding, &closing_txid, their_feerange); if (our_feerange && *their_feerange) { do_quickclose(offer, pps, &channel_id, funding_pubkey, funding_wscript, scriptpubkey, &funding, funding_sats, out, opener, our_dust_limit, wrong_funding, &closing_txid, our_feerange, *their_feerange); goto exit_thru_the_giftshop; } } } /* Now we have first two points, we can init fee range. */ init_feerange(&feerange, max_fee_to_accept, offer); /* Apply (and check) opener offer now. */ adjust_feerange(&feerange, offer[opener], opener); /* Now any extra rounds required. */ while (!amount_sat_eq(offer[LOCAL], offer[REMOTE])) { /* Still don't agree: adjust feerange based on previous offer */ adjust_feerange(&feerange, offer[!whose_turn], !whose_turn); if (whose_turn == LOCAL) { offer[LOCAL] = adjust_offer(pps, &channel_id, &feerange, offer[REMOTE], min_fee_to_accept, fee_negotiation_step, fee_negotiation_step_unit); send_offer(pps, chainparams, &channel_id, funding_pubkey, funding_wscript, scriptpubkey, &funding, funding_sats, out, opener, our_dust_limit, offer[LOCAL], wrong_funding, our_feerange); } else { peer_billboard(false, "Waiting for another" " closing fee offer:" " ours was %"PRIu64" satoshi," " theirs was %"PRIu64" satoshi,", offer[LOCAL], offer[REMOTE]); offer[REMOTE] = receive_offer(pps, chainparams, &channel_id, funding_pubkey, funding_wscript, scriptpubkey, &funding, funding_sats, out, opener, our_dust_limit, min_fee_to_accept, wrong_funding, &closing_txid, their_feerange); } whose_turn = !whose_turn; } peer_billboard(true, "We agreed on a closing fee of %"PRIu64" satoshi for tx:%s", offer[LOCAL], type_to_string(tmpctx, struct bitcoin_txid, &closing_txid)); exit_thru_the_giftshop: #if DEVELOPER /* We don't listen for master commands, so always check memleak here */ tal_free(wrong_funding); tal_free(our_feerange); tal_free(their_feerange); tal_free(max_feerate); closing_dev_memleak(ctx, scriptpubkey, funding_wscript); #endif /* We're done! */ /* Properly close the channel first. */ if (!socket_close(pps->peer_fd)) status_unusual("Closing and draining peerfd gave error: %s", strerror(errno)); /* Sending the below will kill us! */ wire_sync_write(REQ_FD, take(towire_closingd_complete(NULL))); tal_free(ctx); daemon_shutdown(); return 0; }