|
|
|
#include <bitcoin/feerate.h>
|
|
|
|
#include <bitcoin/script.h>
|
|
|
|
#include <common/key_derive.h>
|
|
|
|
#include <common/utils.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#include <inttypes.h>
|
|
|
|
#include <lightningd/chaintopology.h>
|
|
|
|
#include <lightningd/coin_mvts.h>
|
|
|
|
#include <lightningd/hsm_control.h>
|
|
|
|
#include <lightningd/log.h>
|
|
|
|
#include <lightningd/onchain_control.h>
|
|
|
|
#include <lightningd/peer_control.h>
|
|
|
|
#include <lightningd/subd.h>
|
|
|
|
#include <lightningd/watch.h>
|
|
|
|
#include <wire/wire_sync.h>
|
|
|
|
|
|
|
|
/* We dump all the known preimages when onchaind starts up. */
|
|
|
|
static void onchaind_tell_fulfill(struct channel *channel)
|
|
|
|
{
|
|
|
|
struct htlc_in_map_iter ini;
|
|
|
|
struct htlc_in *hin;
|
|
|
|
u8 *msg;
|
|
|
|
struct lightningd *ld = channel->peer->ld;
|
|
|
|
|
|
|
|
for (hin = htlc_in_map_first(&ld->htlcs_in, &ini);
|
|
|
|
hin;
|
|
|
|
hin = htlc_in_map_next(&ld->htlcs_in, &ini)) {
|
|
|
|
if (hin->key.channel != channel)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
/* BOLT #5:
|
|
|
|
*
|
|
|
|
* A local node:
|
|
|
|
|
|
|
|
* - if it receives (or already possesses) a payment preimage
|
|
|
|
* for an unresolved HTLC output that it has been offered AND
|
|
|
|
* for which it has committed to an outgoing HTLC:
|
|
|
|
* - MUST *resolve* the output by spending it, using the
|
|
|
|
* HTLC-success transaction.
|
|
|
|
* - MUST resolve the output of that HTLC-success transaction.
|
|
|
|
* - otherwise:
|
|
|
|
* - if the *remote node* is NOT irrevocably committed to
|
|
|
|
* the HTLC:
|
|
|
|
* - MUST NOT *resolve* the output by spending it.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* We only set preimage once it's irrevocably committed, and
|
|
|
|
* we spend even if we don't have an outgoing HTLC (eg. local
|
|
|
|
* payment complete) */
|
|
|
|
if (!hin->preimage)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
/* Sooo these are *probably* replays since they're coming
|
|
|
|
* from the database but it's hard to be sure since we update
|
|
|
|
* the database before notifying onchaind about them.
|
|
|
|
* There's a *very* rare chance that we'll not log them,
|
|
|
|
* only in that we only make ledger records as a result of this call
|
|
|
|
* iff the output isn't deemed 'trackable'. So if we do miss a
|
|
|
|
* ledger record as a result of this decision, it's guaranteed to be
|
|
|
|
* impreceptibly tiny *and* not show up anywhere else in the node's
|
|
|
|
* utxo set.
|
|
|
|
*
|
|
|
|
* Aka a reconciliator's nightmare.
|
|
|
|
* The alternative is to double-count *every* ignored htlc output
|
|
|
|
* It's easier to delete than find a missing, but I'm banking on
|
|
|
|
* the rarity of failure here. (hahaha) */
|
|
|
|
msg = towire_onchaind_known_preimage(channel, hin->preimage, false);
|
|
|
|
subd_send_msg(channel->owner, take(msg));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void handle_onchain_init_reply(struct channel *channel, const u8 *msg UNUSED)
|
|
|
|
{
|
|
|
|
/* FIXME: We may already be ONCHAIN state when we implement restart! */
|
|
|
|
channel_set_state(channel, FUNDING_SPEND_SEEN, ONCHAIN);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Notify onchaind about the depth change of the watched tx.
|
|
|
|
*/
|
|
|
|
static void onchain_tx_depth(struct channel *channel,
|
|
|
|
const struct bitcoin_txid *txid,
|
|
|
|
unsigned int depth,
|
|
|
|
bool is_replay)
|
|
|
|
{
|
|
|
|
u8 *msg;
|
|
|
|
msg = towire_onchaind_depth(channel, txid, depth, is_replay);
|
|
|
|
subd_send_msg(channel->owner, take(msg));
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Entrypoint for the txwatch callback, calls onchain_tx_depth.
|
|
|
|
*/
|
|
|
|
static enum watch_result onchain_tx_watched(struct lightningd *ld,
|
|
|
|
struct channel *channel,
|
|
|
|
const struct bitcoin_txid *txid,
|
|
|
|
const struct bitcoin_tx *tx,
|
|
|
|
unsigned int depth)
|
|
|
|
{
|
|
|
|
u32 blockheight = get_block_height(ld->topology);
|
|
|
|
|
|
|
|
if (tx != NULL) {
|
|
|
|
struct bitcoin_txid txid2;
|
|
|
|
|
|
|
|
bitcoin_txid(tx, &txid2);
|
|
|
|
if (!bitcoin_txid_eq(txid, &txid2)) {
|
|
|
|
channel_internal_error(channel, "Txid for %s is not %s",
|
|
|
|
type_to_string(tmpctx,
|
|
|
|
struct bitcoin_tx,
|
|
|
|
tx),
|
|
|
|
type_to_string(tmpctx,
|
|
|
|
struct bitcoin_txid,
|
|
|
|
txid));
|
|
|
|
return DELETE_WATCH;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (depth == 0) {
|
|
|
|
log_unusual(channel->log, "Chain reorganization!");
|
|
|
|
channel_set_owner(channel, NULL);
|
|
|
|
|
|
|
|
/* We will most likely be freed, so this is a noop */
|
|
|
|
return KEEP_WATCHING;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Store the channeltx so we can replay later */
|
|
|
|
wallet_channeltxs_add(ld->wallet, channel,
|
|
|
|
WIRE_ONCHAIND_DEPTH, txid, 0, blockheight);
|
|
|
|
|
|
|
|
onchain_tx_depth(channel, txid, depth, false);
|
|
|
|
return KEEP_WATCHING;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void watch_tx_and_outputs(struct channel *channel,
|
|
|
|
const struct bitcoin_tx *tx);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Notify onchaind that an output was spent and register new watches.
|
|
|
|
*/
|
|
|
|
static void onchain_txo_spent(struct channel *channel, const struct bitcoin_tx *tx, size_t input_num, u32 blockheight, bool is_replay)
|
|
|
|
{
|
|
|
|
u8 *msg;
|
|
|
|
/* Onchaind needs all inputs, since it uses those to compare
|
|
|
|
* with existing spends (which can vary, with feerate changes). */
|
|
|
|
struct tx_parts *parts = tx_parts_from_wally_tx(tmpctx, tx->wtx,
|
|
|
|
-1, -1);
|
|
|
|
|
|
|
|
watch_tx_and_outputs(channel, tx);
|
|
|
|
|
|
|
|
msg = towire_onchaind_spent(channel, parts, input_num, blockheight, is_replay);
|
|
|
|
subd_send_msg(channel->owner, take(msg));
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Entrypoint for the txowatch callback, stores tx and calls onchain_txo_spent.
|
|
|
|
*/
|
|
|
|
static enum watch_result onchain_txo_watched(struct channel *channel,
|
|
|
|
const struct bitcoin_tx *tx,
|
|
|
|
size_t input_num,
|
|
|
|
const struct block *block)
|
|
|
|
{
|
|
|
|
struct bitcoin_txid txid;
|
|
|
|
bitcoin_txid(tx, &txid);
|
|
|
|
|
|
|
|
/* Store the channeltx so we can replay later */
|
|
|
|
wallet_channeltxs_add(channel->peer->ld->wallet, channel,
|
|
|
|
WIRE_ONCHAIND_SPENT, &txid, input_num,
|
|
|
|
block->height);
|
|
|
|
|
|
|
|
onchain_txo_spent(channel, tx, input_num, block->height, false);
|
|
|
|
|
|
|
|
/* We don't need to keep watching: If this output is double-spent
|
|
|
|
* (reorg), we'll get a zero depth cb to onchain_tx_watched, and
|
|
|
|
* restart onchaind. */
|
|
|
|
return DELETE_WATCH;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* To avoid races, we watch the tx and all outputs. */
|
|
|
|
static void watch_tx_and_outputs(struct channel *channel,
|
|
|
|
const struct bitcoin_tx *tx)
|
|
|
|
{
|
|
|
|
struct bitcoin_txid txid;
|
|
|
|
struct txwatch *txw;
|
|
|
|
struct lightningd *ld = channel->peer->ld;
|
|
|
|
|
|
|
|
bitcoin_txid(tx, &txid);
|
|
|
|
|
|
|
|
/* Make txwatch a parent of txo watches, so we can unwatch together. */
|
|
|
|
txw = watch_tx(channel->owner, ld->topology, channel, tx,
|
|
|
|
onchain_tx_watched);
|
|
|
|
|
|
|
|
for (size_t i = 0; i < tx->wtx->num_outputs; i++)
|
|
|
|
watch_txo(txw, ld->topology, channel, &txid, i,
|
|
|
|
onchain_txo_watched);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void handle_onchain_log_coin_move(struct channel *channel, const u8 *msg)
|
|
|
|
{
|
|
|
|
struct channel_id channel_id;
|
|
|
|
struct chain_coin_mvt *mvt = tal(NULL, struct chain_coin_mvt);
|
|
|
|
|
|
|
|
if (!fromwire_onchaind_notify_coin_mvt(msg, mvt)) {
|
|
|
|
channel_internal_error(channel, "Invalid onchain notify_coin_mvt");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
derive_channel_id(&channel_id, &channel->funding_txid,
|
|
|
|
channel->funding_outnum);
|
|
|
|
mvt->account_name =
|
|
|
|
type_to_string(mvt, struct channel_id, &channel_id);
|
|
|
|
notify_chain_mvt(channel->peer->ld, mvt);
|
|
|
|
tal_free(mvt);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void handle_onchain_broadcast_tx(struct channel *channel, const u8 *msg)
|
|
|
|
{
|
|
|
|
struct bitcoin_tx *tx;
|
|
|
|
struct wallet *w = channel->peer->ld->wallet;
|
|
|
|
struct bitcoin_txid txid;
|
|
|
|
enum wallet_tx_type type;
|
|
|
|
|
|
|
|
if (!fromwire_onchaind_broadcast_tx(msg, msg, &tx, &type)) {
|
|
|
|
channel_internal_error(channel, "Invalid onchain_broadcast_tx");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
tx->chainparams = chainparams;
|
|
|
|
|
|
|
|
bitcoin_txid(tx, &txid);
|
|
|
|
wallet_transaction_add(w, tx->wtx, 0, 0);
|
|
|
|
wallet_transaction_annotate(w, &txid, type, channel->dbid);
|
|
|
|
|
|
|
|
/* We don't really care if it fails, we'll respond via watch. */
|
|
|
|
broadcast_tx(channel->peer->ld->topology, channel, tx, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void handle_onchain_unwatch_tx(struct channel *channel, const u8 *msg)
|
|
|
|
{
|
|
|
|
struct bitcoin_txid txid;
|
|
|
|
struct txwatch *txw;
|
|
|
|
|
|
|
|
if (!fromwire_onchaind_unwatch_tx(msg, &txid)) {
|
|
|
|
channel_internal_error(channel, "Invalid onchain_unwatch_tx");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Frees the txo watches, too: see watch_tx_and_outputs() */
|
|
|
|
txw = find_txwatch(channel->peer->ld->topology, &txid, channel);
|
|
|
|
if (!txw)
|
|
|
|
log_unusual(channel->log, "Can't unwatch txid %s",
|
|
|
|
type_to_string(tmpctx, struct bitcoin_txid, &txid));
|
|
|
|
tal_free(txw);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void handle_extracted_preimage(struct channel *channel, const u8 *msg)
|
|
|
|
{
|
|
|
|
struct preimage preimage;
|
|
|
|
|
|
|
|
if (!fromwire_onchaind_extracted_preimage(msg, &preimage)) {
|
|
|
|
channel_internal_error(channel, "Invalid extracted_preimage");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
onchain_fulfilled_htlc(channel, &preimage);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void handle_missing_htlc_output(struct channel *channel, const u8 *msg)
|
|
|
|
{
|
|
|
|
struct htlc_stub htlc;
|
|
|
|
|
|
|
|
if (!fromwire_onchaind_missing_htlc_output(msg, &htlc)) {
|
|
|
|
channel_internal_error(channel, "Invalid missing_htlc_output");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* BOLT #5:
|
|
|
|
*
|
|
|
|
* - for any committed HTLC that does NOT have an output in this
|
|
|
|
* commitment transaction:
|
|
|
|
* - once the commitment transaction has reached reasonable depth:
|
|
|
|
* - MUST fail the corresponding incoming HTLC (if any).
|
|
|
|
* - if no *valid* commitment transaction contains an output
|
|
|
|
* corresponding to the HTLC.
|
|
|
|
* - MAY fail the corresponding incoming HTLC sooner.
|
|
|
|
*/
|
|
|
|
onchain_failed_our_htlc(channel, &htlc, "missing in commitment tx");
|
|
|
|
}
|
|
|
|
|
|
|
|
static void handle_onchain_htlc_timeout(struct channel *channel, const u8 *msg)
|
|
|
|
{
|
|
|
|
struct htlc_stub htlc;
|
|
|
|
|
|
|
|
if (!fromwire_onchaind_htlc_timeout(msg, &htlc)) {
|
|
|
|
channel_internal_error(channel, "Invalid onchain_htlc_timeout");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* BOLT #5:
|
|
|
|
*
|
|
|
|
* - if the commitment transaction HTLC output has *timed out* and
|
|
|
|
* hasn't been *resolved*:
|
|
|
|
* - MUST *resolve* the output by spending it using the HTLC-timeout
|
|
|
|
* transaction.
|
|
|
|
*/
|
|
|
|
onchain_failed_our_htlc(channel, &htlc, "timed out");
|
|
|
|
}
|
|
|
|
|
|
|
|
static void handle_irrevocably_resolved(struct channel *channel, const u8 *msg UNUSED)
|
|
|
|
{
|
|
|
|
/* FIXME: Implement check_htlcs to ensure no dangling hout->in ptrs! */
|
|
|
|
free_htlcs(channel->peer->ld, channel);
|
|
|
|
|
|
|
|
log_info(channel->log, "onchaind complete, forgetting peer");
|
|
|
|
|
|
|
|
/* This will also free onchaind. */
|
|
|
|
delete_channel(channel);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* onchain_add_utxo -- onchaind is telling us about an UTXO we own
|
|
|
|
*/
|
|
|
|
static void onchain_add_utxo(struct channel *channel, const u8 *msg)
|
|
|
|
{
|
|
|
|
struct chain_coin_mvt *mvt;
|
|
|
|
u32 blockheight;
|
|
|
|
struct bitcoin_txid txid;
|
|
|
|
u32 outnum;
|
|
|
|
struct amount_sat amount;
|
|
|
|
struct pubkey *commitment_point;
|
|
|
|
u8 *scriptPubkey;
|
|
|
|
|
|
|
|
if (!fromwire_onchaind_add_utxo(
|
|
|
|
tmpctx, msg, &txid, &outnum, &commitment_point,
|
|
|
|
&amount, &blockheight, &scriptPubkey)) {
|
|
|
|
log_broken(channel->log,
|
|
|
|
"onchaind gave invalid add_utxo message: %s",
|
|
|
|
tal_hex(msg, msg));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
assert(blockheight);
|
|
|
|
outpointfilter_add(channel->peer->ld->wallet->owned_outpoints,
|
|
|
|
&txid, outnum);
|
|
|
|
wallet_add_onchaind_utxo(channel->peer->ld->wallet,
|
|
|
|
&txid, outnum, scriptPubkey,
|
|
|
|
blockheight, amount, channel,
|
|
|
|
commitment_point);
|
|
|
|
|
|
|
|
mvt = new_coin_deposit_sat(msg, "wallet", &txid,
|
|
|
|
outnum, blockheight, amount);
|
|
|
|
notify_chain_mvt(channel->peer->ld, mvt);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void onchain_annotate_txout(struct channel *channel, const u8 *msg)
|
|
|
|
{
|
|
|
|
struct bitcoin_txid txid;
|
|
|
|
enum wallet_tx_type type;
|
|
|
|
u32 outnum;
|
|
|
|
if (!fromwire_onchaind_annotate_txout(msg, &txid, &outnum, &type))
|
|
|
|
fatal("onchaind gave invalid onchain_annotate_txout "
|
|
|
|
"message: %s",
|
|
|
|
tal_hex(msg, msg));
|
|
|
|
wallet_annotate_txout(channel->peer->ld->wallet, &txid, outnum, type,
|
|
|
|
channel->dbid);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void onchain_annotate_txin(struct channel *channel, const u8 *msg)
|
|
|
|
{
|
|
|
|
struct bitcoin_txid txid;
|
|
|
|
enum wallet_tx_type type;
|
|
|
|
u32 innum;
|
|
|
|
if (!fromwire_onchaind_annotate_txin(msg, &txid, &innum, &type))
|
|
|
|
fatal("onchaind gave invalid onchain_annotate_txin "
|
|
|
|
"message: %s",
|
|
|
|
tal_hex(msg, msg));
|
|
|
|
wallet_annotate_txin(channel->peer->ld->wallet, &txid, innum, type,
|
|
|
|
channel->dbid);
|
|
|
|
}
|
|
|
|
|
|
|
|
static unsigned int onchain_msg(struct subd *sd, const u8 *msg, const int *fds UNUSED)
|
|
|
|
{
|
|
|
|
enum onchaind_wire t = fromwire_peektype(msg);
|
|
|
|
|
|
|
|
switch (t) {
|
|
|
|
case WIRE_ONCHAIND_INIT_REPLY:
|
|
|
|
handle_onchain_init_reply(sd->channel, msg);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case WIRE_ONCHAIND_BROADCAST_TX:
|
|
|
|
handle_onchain_broadcast_tx(sd->channel, msg);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case WIRE_ONCHAIND_UNWATCH_TX:
|
|
|
|
handle_onchain_unwatch_tx(sd->channel, msg);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case WIRE_ONCHAIND_EXTRACTED_PREIMAGE:
|
|
|
|
handle_extracted_preimage(sd->channel, msg);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case WIRE_ONCHAIND_MISSING_HTLC_OUTPUT:
|
|
|
|
handle_missing_htlc_output(sd->channel, msg);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case WIRE_ONCHAIND_HTLC_TIMEOUT:
|
|
|
|
handle_onchain_htlc_timeout(sd->channel, msg);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case WIRE_ONCHAIND_ALL_IRREVOCABLY_RESOLVED:
|
|
|
|
handle_irrevocably_resolved(sd->channel, msg);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case WIRE_ONCHAIND_ADD_UTXO:
|
|
|
|
onchain_add_utxo(sd->channel, msg);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case WIRE_ONCHAIND_ANNOTATE_TXIN:
|
|
|
|
onchain_annotate_txin(sd->channel, msg);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case WIRE_ONCHAIND_ANNOTATE_TXOUT:
|
|
|
|
onchain_annotate_txout(sd->channel, msg);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case WIRE_ONCHAIND_NOTIFY_COIN_MVT:
|
|
|
|
handle_onchain_log_coin_move(sd->channel, msg);
|
|
|
|
break;
|
|
|
|
|
|
|
|
/* We send these, not receive them */
|
|
|
|
case WIRE_ONCHAIND_INIT:
|
|
|
|
case WIRE_ONCHAIND_SPENT:
|
|
|
|
case WIRE_ONCHAIND_DEPTH:
|
|
|
|
case WIRE_ONCHAIND_HTLC:
|
|
|
|
case WIRE_ONCHAIND_KNOWN_PREIMAGE:
|
|
|
|
case WIRE_ONCHAIND_DEV_MEMLEAK:
|
|
|
|
case WIRE_ONCHAIND_DEV_MEMLEAK_REPLY:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* If we want to know if this HTLC is missing, return depth. */
|
|
|
|
static bool tell_if_missing(const struct channel *channel,
|
|
|
|
struct htlc_stub *stub,
|
|
|
|
bool *tell_immediate)
|
|
|
|
{
|
|
|
|
struct htlc_out *hout;
|
|
|
|
|
|
|
|
/* Keep valgrind happy. */
|
|
|
|
*tell_immediate = false;
|
|
|
|
|
|
|
|
/* Don't care about incoming HTLCs, just ones we offered. */
|
|
|
|
if (stub->owner == REMOTE)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
/* Might not be a current HTLC. */
|
|
|
|
hout = find_htlc_out(&channel->peer->ld->htlcs_out, channel, stub->id);
|
|
|
|
if (!hout)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
/* BOLT #5:
|
|
|
|
*
|
|
|
|
* - for any committed HTLC that does NOT have an output in this
|
|
|
|
* commitment transaction:
|
|
|
|
* - once the commitment transaction has reached reasonable depth:
|
|
|
|
* - MUST fail the corresponding incoming HTLC (if any).
|
|
|
|
* - if no *valid* commitment transaction contains an output
|
|
|
|
* corresponding to the HTLC.
|
|
|
|
* - MAY fail the corresponding incoming HTLC sooner.
|
|
|
|
*/
|
|
|
|
if (hout->hstate >= RCVD_ADD_REVOCATION
|
|
|
|
&& hout->hstate < SENT_REMOVE_REVOCATION)
|
|
|
|
*tell_immediate = true;
|
|
|
|
|
|
|
|
log_debug(channel->log,
|
|
|
|
"We want to know if htlc %"PRIu64" is missing (%s)",
|
|
|
|
hout->key.id, *tell_immediate ? "immediate" : "later");
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Only error onchaind can get is if it dies. */
|
|
|
|
static void onchain_error(struct channel *channel,
|
|
|
|
struct per_peer_state *pps UNUSED,
|
|
|
|
const struct channel_id *channel_id UNUSED,
|
|
|
|
const char *desc,
|
|
|
|
bool soft_error UNUSED,
|
|
|
|
const u8 *err_for_them UNUSED)
|
|
|
|
{
|
|
|
|
/* FIXME: re-launch? */
|
|
|
|
log_broken(channel->log, "%s", desc);
|
|
|
|
channel_set_billboard(channel, true, desc);
|
|
|
|
channel_set_owner(channel, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* With a reorg, this can get called multiple times; each time we'll kill
|
|
|
|
* onchaind (like any other owner), and restart */
|
|
|
|
enum watch_result onchaind_funding_spent(struct channel *channel,
|
|
|
|
const struct bitcoin_tx *tx,
|
|
|
|
u32 blockheight,
|
|
|
|
bool is_replay)
|
|
|
|
{
|
|
|
|
u8 *msg;
|
|
|
|
struct bitcoin_txid our_last_txid;
|
|
|
|
struct htlc_stub *stubs;
|
|
|
|
struct lightningd *ld = channel->peer->ld;
|
|
|
|
struct pubkey final_key;
|
|
|
|
int hsmfd;
|
|
|
|
u32 feerates[3];
|
|
|
|
|
|
|
|
channel_fail_permanent(channel, "Funding transaction spent");
|
|
|
|
|
|
|
|
/* We could come from almost any state. */
|
|
|
|
channel_set_state(channel, channel->state, FUNDING_SPEND_SEEN);
|
|
|
|
|
|
|
|
hsmfd = hsm_get_client_fd(ld, &channel->peer->id,
|
|
|
|
channel->dbid,
|
|
|
|
HSM_CAP_SIGN_ONCHAIN_TX
|
|
|
|
| HSM_CAP_COMMITMENT_POINT);
|
|
|
|
|
|
|
|
channel_set_owner(channel, new_channel_subd(ld,
|
|
|
|
"lightning_onchaind",
|
|
|
|
channel, &channel->peer->id,
|
|
|
|
channel->log, false,
|
|
|
|
onchaind_wire_name,
|
|
|
|
onchain_msg,
|
|
|
|
onchain_error,
|
|
|
|
channel_set_billboard,
|
|
|
|
take(&hsmfd),
|
|
|
|
NULL));
|
|
|
|
|
|
|
|
if (!channel->owner) {
|
|
|
|
log_broken(channel->log, "Could not subdaemon onchain: %s",
|
|
|
|
strerror(errno));
|
|
|
|
return KEEP_WATCHING;
|
|
|
|
}
|
|
|
|
|
|
|
|
stubs = wallet_htlc_stubs(tmpctx, ld->wallet, channel);
|
|
|
|
if (!stubs) {
|
|
|
|
log_broken(channel->log, "Could not load htlc_stubs");
|
|
|
|
return KEEP_WATCHING;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!bip32_pubkey(ld->wallet->bip32_base, &final_key,
|
|
|
|
channel->final_key_idx)) {
|
|
|
|
log_broken(channel->log, "Could not derive onchain key %"PRIu64,
|
|
|
|
channel->final_key_idx);
|
|
|
|
return KEEP_WATCHING;
|
|
|
|
}
|
|
|
|
/* This could be a mutual close, but it doesn't matter. */
|
|
|
|
bitcoin_txid(channel->last_tx, &our_last_txid);
|
|
|
|
|
|
|
|
/* We try to get the feerate for each transaction type, 0 if estimation
|
|
|
|
* failed. */
|
|
|
|
feerates[0] = delayed_to_us_feerate(ld->topology);
|
|
|
|
feerates[1] = htlc_resolution_feerate(ld->topology);
|
|
|
|
feerates[2] = penalty_feerate(ld->topology);
|
|
|
|
/* We check them separately but there is a high chance that if estimation
|
|
|
|
* failed for one, it failed for all.. */
|
|
|
|
for (size_t i = 0; i < 3; i++) {
|
|
|
|
if (!feerates[i]) {
|
|
|
|
/* We have at least one data point: the last tx's feerate. */
|
|
|
|
struct amount_sat fee = channel->funding;
|
|
|
|
for (size_t i = 0;
|
|
|
|
i < channel->last_tx->wtx->num_outputs; i++) {
|
|
|
|
struct amount_asset asset =
|
|
|
|
bitcoin_tx_output_get_amount(channel->last_tx, i);
|
|
|
|
struct amount_sat amt;
|
|
|
|
assert(amount_asset_is_main(&asset));
|
|
|
|
amt = amount_asset_to_sat(&asset);
|
|
|
|
if (!amount_sat_sub(&fee, fee, amt)) {
|
|
|
|
log_broken(channel->log, "Could not get fee"
|
|
|
|
" funding %s tx %s",
|
|
|
|
type_to_string(tmpctx,
|
|
|
|
struct amount_sat,
|
|
|
|
&channel->funding),
|
|
|
|
type_to_string(tmpctx,
|
|
|
|
struct bitcoin_tx,
|
|
|
|
channel->last_tx));
|
|
|
|
return KEEP_WATCHING;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
feerates[i] = fee.satoshis / bitcoin_tx_weight(tx); /* Raw: reverse feerate extraction */
|
|
|
|
if (feerates[i] < feerate_floor())
|
|
|
|
feerates[i] = feerate_floor();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
msg = towire_onchaind_init(channel,
|
|
|
|
&channel->their_shachain.chain,
|
|
|
|
chainparams,
|
|
|
|
channel->funding,
|
|
|
|
channel->our_msat,
|
|
|
|
&channel->channel_info.old_remote_per_commit,
|
|
|
|
&channel->channel_info.remote_per_commit,
|
|
|
|
/* BOLT #2:
|
|
|
|
* `to_self_delay` is the number of blocks
|
|
|
|
* that the other node's to-self outputs
|
|
|
|
* must be delayed */
|
|
|
|
/* So, these are reversed: they specify ours,
|
|
|
|
* we specify theirs. */
|
|
|
|
channel->channel_info.their_config.to_self_delay,
|
|
|
|
channel->our_config.to_self_delay,
|
|
|
|
/* delayed_to_us, htlc, and penalty. */
|
|
|
|
feerates[0], feerates[1], feerates[2],
|
|
|
|
channel->our_config.dust_limit,
|
|
|
|
&our_last_txid,
|
|
|
|
channel->shutdown_scriptpubkey[LOCAL],
|
|
|
|
channel->shutdown_scriptpubkey[REMOTE],
|
|
|
|
&final_key,
|
|
|
|
channel->opener,
|
|
|
|
&channel->local_basepoints,
|
|
|
|
&channel->channel_info.theirbase,
|
|
|
|
tx_parts_from_wally_tx(tmpctx, tx->wtx, -1, -1),
|
|
|
|
tx->wtx->locktime,
|
|
|
|
blockheight,
|
|
|
|
/* FIXME: config for 'reasonable depth' */
|
|
|
|
3,
|
|
|
|
channel->last_htlc_sigs,
|
|
|
|
tal_count(stubs),
|
|
|
|
channel->min_possible_feerate,
|
|
|
|
channel->max_possible_feerate,
|
|
|
|
channel->future_per_commitment_point,
|
|
|
|
&channel->local_funding_pubkey,
|
|
|
|
&channel->channel_info.remote_fundingkey,
|
|
|
|
channel->option_static_remotekey,
|
|
|
|
channel->option_anchor_outputs,
|
|
|
|
is_replay);
|
|
|
|
subd_send_msg(channel->owner, take(msg));
|
|
|
|
|
|
|
|
/* FIXME: Don't queue all at once, use an empty cb... */
|
|
|
|
for (size_t i = 0; i < tal_count(stubs); i++) {
|
|
|
|
bool tell_immediate;
|
|
|
|
bool tell = tell_if_missing(channel, &stubs[i], &tell_immediate);
|
|
|
|
msg = towire_onchaind_htlc(channel, &stubs[i],
|
|
|
|
tell, tell_immediate);
|
|
|
|
subd_send_msg(channel->owner, take(msg));
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Tell it about any preimages we know. */
|
|
|
|
onchaind_tell_fulfill(channel);
|
|
|
|
|
|
|
|
watch_tx_and_outputs(channel, tx);
|
|
|
|
|
|
|
|
/* We keep watching until peer finally deleted, for reorgs. */
|
|
|
|
return KEEP_WATCHING;
|
|
|
|
}
|
|
|
|
|
|
|
|
void onchaind_replay_channels(struct lightningd *ld)
|
|
|
|
{
|
|
|
|
u32 *onchaind_ids;
|
|
|
|
struct channeltx *txs;
|
|
|
|
struct channel *chan;
|
|
|
|
|
|
|
|
db_begin_transaction(ld->wallet->db);
|
|
|
|
onchaind_ids = wallet_onchaind_channels(ld->wallet, ld);
|
|
|
|
|
|
|
|
for (size_t i = 0; i < tal_count(onchaind_ids); i++) {
|
|
|
|
log_info(ld->log, "Restarting onchaind for channel %d",
|
|
|
|
onchaind_ids[i]);
|
|
|
|
|
|
|
|
txs = wallet_channeltxs_get(ld->wallet, onchaind_ids,
|
|
|
|
onchaind_ids[i]);
|
|
|
|
chan = channel_by_dbid(ld, onchaind_ids[i]);
|
|
|
|
|
|
|
|
for (size_t j = 0; j < tal_count(txs); j++) {
|
|
|
|
if (txs[j].type == WIRE_ONCHAIND_INIT) {
|
|
|
|
onchaind_funding_spent(chan, txs[j].tx,
|
|
|
|
txs[j].blockheight,
|
|
|
|
true);
|
|
|
|
|
|
|
|
} else if (txs[j].type == WIRE_ONCHAIND_SPENT) {
|
|
|
|
onchain_txo_spent(chan, txs[j].tx,
|
|
|
|
txs[j].input_num,
|
|
|
|
txs[j].blockheight,
|
|
|
|
true);
|
|
|
|
|
|
|
|
} else if (txs[j].type == WIRE_ONCHAIND_DEPTH) {
|
|
|
|
onchain_tx_depth(chan, &txs[j].txid,
|
|
|
|
txs[j].depth, true);
|
|
|
|
|
|
|
|
} else {
|
|
|
|
fatal("unknown message of type %d during "
|
|
|
|
"onchaind replay",
|
|
|
|
txs[j].type);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
tal_free(txs);
|
|
|
|
}
|
|
|
|
tal_free(onchaind_ids);
|
|
|
|
|
|
|
|
db_commit_transaction(ld->wallet->db);
|
|
|
|
}
|