mirror of
https://github.com/ElementsProject/lightning.git
synced 2025-01-08 06:49:31 +01:00
ef28b6112c
This change is really to allow us to have a --dev-fail-on-subdaemon-fail option so we can handle failures from subdaemons generically. It also neatens handling so we can have an explicit callback for "peer did something wrong" (which matters if we want to close the channel in that case). Signed-off-by: Rusty Russell <rusty@rustcorp.com.au>
481 lines
14 KiB
C
481 lines
14 KiB
C
#include <bitcoin/script.h>
|
|
#include <closingd/gen_closing_wire.h>
|
|
#include <common/close_tx.h>
|
|
#include <common/crypto_sync.h>
|
|
#include <common/debug.h>
|
|
#include <common/derive_basepoints.h>
|
|
#include <common/htlc.h>
|
|
#include <common/peer_failed.h>
|
|
#include <common/status.h>
|
|
#include <common/type_to_string.h>
|
|
#include <common/utils.h>
|
|
#include <common/version.h>
|
|
#include <errno.h>
|
|
#include <inttypes.h>
|
|
#include <signal.h>
|
|
#include <stdio.h>
|
|
#include <unistd.h>
|
|
#include <wire/peer_wire.h>
|
|
#include <wire/wire_sync.h>
|
|
|
|
/* stdin == requests, 3 == peer, 4 = gossip */
|
|
#define REQ_FD STDIN_FILENO
|
|
#define PEER_FD 3
|
|
#define GOSSIP_FD 4
|
|
|
|
static struct bitcoin_tx *close_tx(const tal_t *ctx,
|
|
struct crypto_state *cs,
|
|
const struct channel_id *channel_id,
|
|
u8 *scriptpubkey[NUM_SIDES],
|
|
const struct sha256_double *funding_txid,
|
|
unsigned int funding_txout,
|
|
u64 funding_satoshi,
|
|
const u64 satoshi_out[NUM_SIDES],
|
|
enum side funder,
|
|
uint64_t fee,
|
|
uint64_t dust_limit)
|
|
{
|
|
struct bitcoin_tx *tx;
|
|
|
|
if (satoshi_out[funder] < fee)
|
|
peer_failed(PEER_FD, cs, channel_id,
|
|
"Funder cannot afford fee %"PRIu64
|
|
" (%"PRIu64" and %"PRIu64")",
|
|
fee, satoshi_out[LOCAL],
|
|
satoshi_out[REMOTE]);
|
|
|
|
status_trace("Making close tx at = %"PRIu64"/%"PRIu64" fee %"PRIu64,
|
|
satoshi_out[LOCAL], satoshi_out[REMOTE], fee);
|
|
|
|
/* FIXME: We need to allow this! */
|
|
tx = create_close_tx(ctx,
|
|
scriptpubkey[LOCAL], scriptpubkey[REMOTE],
|
|
funding_txid,
|
|
funding_txout,
|
|
funding_satoshi,
|
|
satoshi_out[LOCAL] - (funder == LOCAL ? fee : 0),
|
|
satoshi_out[REMOTE] - (funder == REMOTE ? fee : 0),
|
|
dust_limit);
|
|
if (!tx)
|
|
peer_failed(PEER_FD, cs, channel_id,
|
|
"Both outputs below dust limit:"
|
|
" funding = %"PRIu64
|
|
" fee = %"PRIu64
|
|
" dust_limit = %"PRIu64
|
|
" LOCAL = %"PRIu64
|
|
" REMOTE = %"PRIu64,
|
|
funding_satoshi,
|
|
fee,
|
|
dust_limit,
|
|
satoshi_out[LOCAL],
|
|
satoshi_out[REMOTE]);
|
|
return tx;
|
|
}
|
|
|
|
static u64 one_towards(u64 target, u64 value)
|
|
{
|
|
if (value > target)
|
|
return value-1;
|
|
else if (value < target)
|
|
return value+1;
|
|
return value;
|
|
}
|
|
|
|
static void do_reconnect(struct crypto_state *cs,
|
|
const struct channel_id *channel_id,
|
|
const u64 next_index[NUM_SIDES],
|
|
u64 revocations_received)
|
|
{
|
|
u8 *msg;
|
|
struct channel_id their_channel_id;
|
|
const tal_t *tmpctx = tal_tmpctx(NULL);
|
|
u64 next_local_commitment_number, next_remote_revocation_number;
|
|
|
|
/* BOLT #2:
|
|
*
|
|
* On reconnection, a node MUST transmit `channel_reestablish` for
|
|
* each channel, and MUST wait for to receive the other node's
|
|
* `channel_reestablish` message before sending any other messages for
|
|
* that channel. The sending node MUST set
|
|
* `next_local_commitment_number` to the commitment number of the next
|
|
* `commitment_signed` it expects to receive, and MUST set
|
|
* `next_remote_revocation_number` to the commitment number of the
|
|
* next `revoke_and_ack` message it expects to receive.
|
|
*/
|
|
msg = towire_channel_reestablish(tmpctx, channel_id,
|
|
next_index[LOCAL],
|
|
revocations_received);
|
|
if (!sync_crypto_write(cs, PEER_FD, take(msg)))
|
|
status_failed(STATUS_FAIL_PEER_IO,
|
|
"Failed writing reestablish: %s", strerror(errno));
|
|
|
|
again:
|
|
msg = sync_crypto_read(tmpctx, cs, PEER_FD);
|
|
if (!msg)
|
|
status_failed(STATUS_FAIL_PEER_IO,
|
|
"Failed reading reestablish: %s", strerror(errno));
|
|
|
|
if (is_gossip_msg(msg)) {
|
|
if (!wire_sync_write(GOSSIP_FD, take(msg)))
|
|
status_failed(STATUS_FAIL_GOSSIP_IO, "Writing gossip");
|
|
goto again;
|
|
}
|
|
|
|
if (!fromwire_channel_reestablish(msg, NULL, &their_channel_id,
|
|
&next_local_commitment_number,
|
|
&next_remote_revocation_number)) {
|
|
peer_failed(PEER_FD, cs, channel_id,
|
|
"bad reestablish msg: %s %s",
|
|
wire_type_name(fromwire_peektype(msg)),
|
|
tal_hex(tmpctx, msg));
|
|
}
|
|
status_trace("Got reestablish commit=%"PRIu64" revoke=%"PRIu64,
|
|
next_local_commitment_number,
|
|
next_remote_revocation_number);
|
|
|
|
/* FIXME: Spec says to re-xmit funding_locked here if we haven't
|
|
* done any updates. */
|
|
|
|
/* BOLT #2:
|
|
*
|
|
* On reconnection if the node has sent a previous `closing_signed` it
|
|
* MUST send another `closing_signed`
|
|
*/
|
|
|
|
/* Since we always transmit closing_signed immediately, if
|
|
* we're reconnecting we consider ourselves to have transmitted once,
|
|
* and we'll immediately do the retransmit now anyway. */
|
|
tal_free(tmpctx);
|
|
}
|
|
|
|
int main(int argc, char *argv[])
|
|
{
|
|
struct crypto_state cs;
|
|
const tal_t *ctx = tal_tmpctx(NULL);
|
|
u8 *msg;
|
|
struct privkey seed;
|
|
struct pubkey funding_pubkey[NUM_SIDES];
|
|
struct sha256_double funding_txid;
|
|
u16 funding_txout;
|
|
u64 funding_satoshi, satoshi_out[NUM_SIDES];
|
|
u64 our_dust_limit;
|
|
u64 minfee, maxfee, sent_fee;
|
|
s64 last_received_fee = -1;
|
|
enum side funder;
|
|
u8 *scriptpubkey[NUM_SIDES], *funding_wscript;
|
|
struct channel_id channel_id;
|
|
struct secrets secrets;
|
|
secp256k1_ecdsa_signature sig;
|
|
bool reconnected;
|
|
u64 next_index[NUM_SIDES], revocations_received;
|
|
|
|
if (argc == 2 && streq(argv[1], "--version")) {
|
|
printf("%s\n", version());
|
|
exit(0);
|
|
}
|
|
|
|
subdaemon_debug(argc, argv);
|
|
|
|
/* We handle write returning errors! */
|
|
signal(SIGCHLD, SIG_IGN);
|
|
secp256k1_ctx = secp256k1_context_create(SECP256K1_CONTEXT_VERIFY
|
|
| SECP256K1_CONTEXT_SIGN);
|
|
status_setup_sync(REQ_FD);
|
|
|
|
msg = wire_sync_read(ctx, REQ_FD);
|
|
if (!fromwire_closing_init(ctx, msg, NULL,
|
|
&cs, &seed,
|
|
&funding_txid, &funding_txout,
|
|
&funding_satoshi,
|
|
&funding_pubkey[REMOTE],
|
|
&funder,
|
|
&satoshi_out[LOCAL],
|
|
&satoshi_out[REMOTE],
|
|
&our_dust_limit,
|
|
&minfee, &maxfee, &sent_fee,
|
|
&scriptpubkey[LOCAL],
|
|
&scriptpubkey[REMOTE],
|
|
&reconnected,
|
|
&next_index[LOCAL],
|
|
&next_index[REMOTE],
|
|
&revocations_received))
|
|
master_badmsg(WIRE_CLOSING_INIT, msg);
|
|
|
|
status_trace("satoshi_out = %"PRIu64"/%"PRIu64,
|
|
satoshi_out[LOCAL], satoshi_out[REMOTE]);
|
|
status_trace("dustlimit = %"PRIu64, our_dust_limit);
|
|
status_trace("fee = %"PRIu64, sent_fee);
|
|
derive_channel_id(&channel_id, &funding_txid, funding_txout);
|
|
derive_basepoints(&seed, &funding_pubkey[LOCAL], NULL,
|
|
&secrets, NULL);
|
|
|
|
funding_wscript = bitcoin_redeem_2of2(ctx,
|
|
&funding_pubkey[LOCAL],
|
|
&funding_pubkey[REMOTE]);
|
|
|
|
if (reconnected)
|
|
do_reconnect(&cs, &channel_id, next_index, revocations_received);
|
|
|
|
/* BOLT #2:
|
|
*
|
|
* Nodes SHOULD send a `closing_signed` message after `shutdown` has
|
|
* been received and no HTLCs remain in either commitment transaction.
|
|
*/
|
|
|
|
/* BOLT #2:
|
|
*
|
|
* On reconnection, ... if the node has sent a previous
|
|
* `closing_signed` it MUST send another `closing_signed`, otherwise
|
|
* if the node has sent a previous `shutdown` it MUST retransmit it.
|
|
*/
|
|
for (;;) {
|
|
const tal_t *tmpctx = tal_tmpctx(ctx);
|
|
struct bitcoin_tx *tx;
|
|
u64 received_fee, limit_fee, new_fee;
|
|
|
|
/* BOLT #2:
|
|
*
|
|
* The sender 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, &cs, &channel_id,
|
|
scriptpubkey,
|
|
&funding_txid,
|
|
funding_txout,
|
|
funding_satoshi,
|
|
satoshi_out, funder, sent_fee, our_dust_limit);
|
|
|
|
/* BOLT #3:
|
|
*
|
|
* ## Closing Transaction
|
|
*...
|
|
* Each node offering a signature... MAY also eliminate its
|
|
* own output.
|
|
*/
|
|
/* (We don't do this). */
|
|
sign_tx_input(tx, 0, NULL, funding_wscript,
|
|
&secrets.funding_privkey,
|
|
&funding_pubkey[LOCAL],
|
|
&sig);
|
|
|
|
status_trace("sending fee offer %"PRIu64, sent_fee);
|
|
|
|
/* Now send closing offer */
|
|
msg = towire_closing_signed(tmpctx, &channel_id, sent_fee, &sig);
|
|
if (!sync_crypto_write(&cs, PEER_FD, take(msg)))
|
|
status_failed(STATUS_FAIL_PEER_IO,
|
|
"Writing closing_signed");
|
|
|
|
/* Did we just agree with them? If so, we're done. */
|
|
if (sent_fee == last_received_fee)
|
|
break;
|
|
|
|
again:
|
|
msg = sync_crypto_read(tmpctx, &cs, PEER_FD);
|
|
if (!msg)
|
|
status_failed(STATUS_FAIL_PEER_IO, "Reading input");
|
|
|
|
/* We don't send gossip at this stage, but we can recv it */
|
|
if (is_gossip_msg(msg)) {
|
|
if (!wire_sync_write(GOSSIP_FD, take(msg)))
|
|
status_failed(STATUS_FAIL_GOSSIP_IO,
|
|
"Writing gossip");
|
|
goto again;
|
|
}
|
|
|
|
/* BOLT #2:
|
|
*
|
|
* On reconnection, a node MUST ignore a redundant
|
|
* `funding_locked` if it receives one.
|
|
*/
|
|
/* 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) {
|
|
tal_free(msg);
|
|
goto again;
|
|
}
|
|
|
|
/* BOLT #2:
|
|
*
|
|
* ...if the node has sent a previous `shutdown` it MUST
|
|
* retransmit it.
|
|
*/
|
|
if (fromwire_peektype(msg) == WIRE_SHUTDOWN) {
|
|
tal_free(msg);
|
|
goto again;
|
|
}
|
|
|
|
if (!fromwire_closing_signed(msg, NULL, &channel_id,
|
|
&received_fee, &sig))
|
|
peer_failed(PEER_FD, &cs, &channel_id,
|
|
"Expected closing_signed: %s",
|
|
tal_hex(trc, msg));
|
|
|
|
/* BOLT #2:
|
|
*
|
|
* The receiver MUST check `signature` is valid for either
|
|
* variant of close transaction specified in [BOLT
|
|
* #3](03-transactions.md#closing-transaction), and MUST fail
|
|
* the connection if it is not.
|
|
*/
|
|
tx = close_tx(tmpctx, &cs, &channel_id,
|
|
scriptpubkey,
|
|
&funding_txid,
|
|
funding_txout,
|
|
funding_satoshi,
|
|
satoshi_out, funder, received_fee, our_dust_limit);
|
|
|
|
if (!check_tx_sig(tx, 0, NULL, funding_wscript,
|
|
&funding_pubkey[REMOTE], &sig)) {
|
|
/* Trim it by reducing their output to minimum */
|
|
struct bitcoin_tx *trimmed;
|
|
u64 trimming_satoshi_out[NUM_SIDES];
|
|
|
|
if (funder == REMOTE)
|
|
trimming_satoshi_out[REMOTE] = received_fee;
|
|
else
|
|
trimming_satoshi_out[REMOTE] = 0;
|
|
trimming_satoshi_out[LOCAL] = satoshi_out[LOCAL];
|
|
|
|
/* BOLT #3:
|
|
*
|
|
* Each node offering a signature MUST subtract the
|
|
* fee given by `fee_satoshis` from the output to the
|
|
* funder; it MUST then remove any output below its
|
|
* own `dust_limit_satoshis`, and MAY also eliminate
|
|
* its own output.
|
|
*/
|
|
trimmed = close_tx(tmpctx, &cs, &channel_id,
|
|
scriptpubkey,
|
|
&funding_txid,
|
|
funding_txout,
|
|
funding_satoshi,
|
|
trimming_satoshi_out,
|
|
funder, received_fee, our_dust_limit);
|
|
if (!trimmed
|
|
|| !check_tx_sig(trimmed, 0, NULL, funding_wscript,
|
|
&funding_pubkey[REMOTE], &sig)) {
|
|
peer_failed(PEER_FD, &cs, &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_trace("Received fee offer %"PRIu64, received_fee);
|
|
|
|
/* BOLT #2:
|
|
*
|
|
* Otherwise, the recipient MUST fail the connection if
|
|
* `fee_satoshis` is greater than the base fee of the final
|
|
* commitment transaction as calculated in [BOLT #3] */
|
|
if (received_fee > maxfee)
|
|
peer_failed(PEER_FD, &cs, &channel_id,
|
|
"Bad closing_signed fee %"PRIu64" > %"PRIu64,
|
|
received_fee, maxfee);
|
|
|
|
/* Is fee reasonable? Tell master. */
|
|
if (received_fee < minfee) {
|
|
status_trace("Fee too low, below %"PRIu64, minfee);
|
|
limit_fee = minfee;
|
|
} else {
|
|
status_trace("Fee accepted.");
|
|
msg = towire_closing_received_signature(tmpctx,
|
|
&sig, tx);
|
|
if (!wire_sync_write(REQ_FD, take(msg)))
|
|
status_failed(STATUS_FAIL_MASTER_IO,
|
|
"Writing received to master: %s",
|
|
strerror(errno));
|
|
msg = wire_sync_read(tmpctx, REQ_FD);
|
|
if (!fromwire_closing_received_signature_reply(msg,NULL))
|
|
master_badmsg(WIRE_CLOSING_RECEIVED_SIGNATURE_REPLY,
|
|
msg);
|
|
limit_fee = received_fee;
|
|
}
|
|
|
|
/* BOLT #2:
|
|
*
|
|
* If `fee_satoshis` is equal to its previously sent
|
|
* `fee_satoshis`, the receiver SHOULD sign and broadcast the
|
|
* final closing transaction and MAY close the connection.
|
|
*/
|
|
if (received_fee == sent_fee)
|
|
break;
|
|
|
|
/* BOLT #2:
|
|
*
|
|
* the recipient SHOULD fail the connection if `fee_satoshis`
|
|
* is not strictly between its last-sent `fee_satoshis` and
|
|
* its previously-received `fee_satoshis`, unless it has
|
|
* reconnected since then. */
|
|
if (last_received_fee != -1) {
|
|
bool previous_dir = sent_fee < last_received_fee;
|
|
bool dir = received_fee < last_received_fee;
|
|
bool next_dir = sent_fee < received_fee;
|
|
|
|
/* They went away from our offer? */
|
|
if (dir != previous_dir)
|
|
peer_failed(PEER_FD, &cs, &channel_id,
|
|
"Their fee went %"
|
|
PRIu64" to %"PRIu64
|
|
" when ours was %"PRIu64,
|
|
last_received_fee,
|
|
received_fee,
|
|
sent_fee);
|
|
|
|
/* They jumped over our offer? */
|
|
if (next_dir != previous_dir)
|
|
peer_failed(PEER_FD, &cs, &channel_id,
|
|
"Their fee jumped %"
|
|
PRIu64" to %"PRIu64
|
|
" when ours was %"PRIu64,
|
|
last_received_fee,
|
|
received_fee,
|
|
sent_fee);
|
|
}
|
|
|
|
/* BOLT #2:
|
|
*
|
|
* ...otherwise it MUST propose a value strictly between the
|
|
* received `fee_satoshis` and its previously-sent
|
|
* `fee_satoshis`.
|
|
*/
|
|
|
|
/* We do it by bisection, with twists:
|
|
* 1. Don't go outside limits, or reach them immediately:
|
|
* treat out-of-limit offers as on-limit offers.
|
|
* 2. Round towards the target, otherwise we can't close
|
|
* a final 1-satoshi gap.
|
|
*
|
|
* Note: Overflow impossible here, since fee <= funder amount */
|
|
new_fee = one_towards(limit_fee, limit_fee + sent_fee) / 2;
|
|
|
|
/* If we didn't move, give up (we're ~ at min/max). */
|
|
if (new_fee == sent_fee)
|
|
peer_failed(PEER_FD, &cs, &channel_id,
|
|
"Final fee %"PRIu64" vs %"PRIu64
|
|
" at limits %"PRIu64"-%"PRIu64,
|
|
sent_fee, received_fee,
|
|
minfee, maxfee);
|
|
|
|
last_received_fee = received_fee;
|
|
sent_fee = new_fee;
|
|
tal_free(tmpctx);
|
|
}
|
|
|
|
/* We're done! */
|
|
wire_sync_write(REQ_FD, take(towire_closing_complete(ctx)));
|
|
tal_free(ctx);
|
|
|
|
return 0;
|
|
}
|