Browse Source

peer: handle fulfull and fail onchain.

We now need to use bitcoin_witness_htlc with the r value, so that API
is updated to take 'struct rval' or 'struct sha256'.

We use the nc->delay amount (ie. dstate->config.min_htlc_expiry) to
wait for a timeout refund to be buried before "failing" upstream.
This should probably be made into a clearer parameter rather than
overloading this one.

Signed-off-by: Rusty Russell <rusty@rustcorp.com.au>


Header from folded patch 'dont-use-peer-nc-in-onchain-code.patch':

peer: Don't use peer->nc->delay for onchain case.

Use the config var directly.  We should be freeing peer->nc when the
connection dies anyway.

Signed-off-by: Rusty Russell <rusty@rustcorp.com.au>
ppa-0.6.1
Rusty Russell 9 years ago
parent
commit
caf50c9ae4
  1. 5
      bitcoin/script.c
  2. 2
      bitcoin/script.h
  3. 176
      daemon/peer.c
  4. 15
      daemon/secrets.c
  5. 5
      daemon/secrets.h

5
bitcoin/script.c

@ -487,7 +487,7 @@ u8 **bitcoin_witness_secret(const tal_t *ctx,
} }
u8 **bitcoin_witness_htlc(const tal_t *ctx, u8 **bitcoin_witness_htlc(const tal_t *ctx,
const struct sha256 *htlc_or_revocation_preimage, const void *htlc_or_revocation_preimage,
const struct bitcoin_signature *sig, const struct bitcoin_signature *sig,
const u8 *witnessscript) const u8 *witnessscript)
{ {
@ -498,8 +498,7 @@ u8 **bitcoin_witness_htlc(const tal_t *ctx,
htlc_or_revocation_preimage = &no_preimage; htlc_or_revocation_preimage = &no_preimage;
return bitcoin_witness_secret(ctx, htlc_or_revocation_preimage, return bitcoin_witness_secret(ctx, htlc_or_revocation_preimage,
sizeof(*htlc_or_revocation_preimage), sig, 32, sig, witnessscript);
witnessscript);
} }
bool scripteq(const u8 *s1, size_t s1len, const u8 *s2, size_t s2len) bool scripteq(const u8 *s1, size_t s1len, const u8 *s2, size_t s2len)

2
bitcoin/script.h

@ -89,7 +89,7 @@ u8 **bitcoin_witness_secret(const tal_t *ctx,
/* Create a witness which spends bitcoin_redeeem_htlc_recv/send */ /* Create a witness which spends bitcoin_redeeem_htlc_recv/send */
u8 **bitcoin_witness_htlc(const tal_t *ctx, u8 **bitcoin_witness_htlc(const tal_t *ctx,
const struct sha256 *htlc_or_revocation_preimage, const void *htlc_or_revocation_preimage,
const struct bitcoin_signature *sig, const struct bitcoin_signature *sig,
const u8 *witnessscript); const u8 *witnessscript);

176
daemon/peer.c

@ -547,6 +547,93 @@ static void state_event(struct peer *peer,
} }
} }
static struct htlc *htlc_by_index(const struct commit_info *ci, size_t index)
{
if (ci->map[index] == -1)
return NULL;
/* First two are non-HTLC outputs to us, them. */
assert(index >= 2);
index -= 2;
if (index < tal_count(ci->cstate->side[OURS].htlcs))
return ci->cstate->side[OURS].htlcs[index];
index -= tal_count(ci->cstate->side[OURS].htlcs);
assert(index < tal_count(ci->cstate->side[THEIRS].htlcs));
return ci->cstate->side[THEIRS].htlcs[index];
}
static bool htlc_is_ours(const struct commit_info *ci, size_t index)
{
assert(index >= 2);
index -= 2;
return index < tal_count(ci->cstate->side[OURS].htlcs);
}
/* Create a HTLC fulfill transaction */
static const struct bitcoin_tx *htlc_fulfill_tx(const struct peer *peer,
const struct commit_info *ci,
unsigned int i)
{
u8 *wscript;
struct htlc *htlc;
struct bitcoin_tx *tx = bitcoin_tx(peer, 1, 1);
struct bitcoin_signature sig;
u64 fee, satoshis;
htlc = htlc_by_index(ci, i);
assert(htlc->r);
wscript = bitcoin_redeem_htlc_recv(peer,
&peer->local.finalkey,
&peer->remote.finalkey,
&htlc->expiry,
&peer->remote.locktime,
&ci->revocation_hash,
&htlc->rhash);
tx->input[0].index = ci->map[i];
bitcoin_txid(ci->tx, &tx->input[0].txid);
satoshis = htlc->msatoshis / 1000;
tx->input[0].amount = tal_dup(tx->input, u64, &satoshis);
tx->input[0].sequence_number = bitcoin_nsequence(&peer->remote.locktime);
/* Using a new output address here would be useless: they can tell
* it's their HTLC, and that we collected it via rval. */
tx->output[0].script = scriptpubkey_p2sh(tx,
bitcoin_redeem_single(tx, &peer->local.finalkey));
tx->output[0].script_length = tal_count(tx->output[0].script);
log_debug(peer->log, "Pre-witness txlen = %zu\n",
measure_tx_cost(tx) / 4);
assert(measure_tx_cost(tx) == 83 * 4);
/* Witness length can vary, due to DER encoding of sigs, but we
* use 539 from an example run. */
/* FIXME: Dynamic fees! */
fee = fee_by_feerate(83 + 539 / 4,
peer->dstate->config.closing_fee_rate);
/* FIXME: Fail gracefully in these cases (not worth collecting) */
if (fee > satoshis || is_dust_amount(satoshis - fee))
fatal("HTLC fulfill amount of %"PRIu64" won't cover fee %"PRIu64,
satoshis, fee);
tx->output[0].amount = satoshis - fee;
sig.stype = SIGHASH_ALL;
peer_sign_htlc_fulfill(peer, tx, wscript, &sig.sig);
tx->input[0].witness = bitcoin_witness_htlc(tx, htlc->r, &sig, wscript);
log_debug(peer->log, "tx cost for htlc fulfill tx: %zu",
measure_tx_cost(tx));
return tx;
}
/* FIXME: Reason! */ /* FIXME: Reason! */
static bool command_htlc_fail(struct peer *peer, struct htlc *htlc) static bool command_htlc_fail(struct peer *peer, struct htlc *htlc)
{ {
@ -1231,30 +1318,6 @@ static bool is_mutual_close(const struct peer *peer,
return false; return false;
} }
static struct htlc *htlc_by_index(const struct commit_info *ci, size_t index)
{
if (ci->map[index] == -1)
return NULL;
/* First two are non-HTLC outputs to us, them. */
assert(index >= 2);
index -= 2;
if (index < tal_count(ci->cstate->side[OURS].htlcs))
return ci->cstate->side[OURS].htlcs[index];
index -= tal_count(ci->cstate->side[OURS].htlcs);
assert(index < tal_count(ci->cstate->side[THEIRS].htlcs));
return ci->cstate->side[THEIRS].htlcs[index];
}
static bool htlc_is_ours(const struct commit_info *ci, size_t index)
{
assert(index >= 2);
index -= 2;
return index < tal_count(ci->cstate->side[OURS].htlcs);
}
/* Create a HTLC refund collection */ /* Create a HTLC refund collection */
static const struct bitcoin_tx *htlc_timeout_tx(const struct peer *peer, static const struct bitcoin_tx *htlc_timeout_tx(const struct peer *peer,
const struct commit_info *ci, const struct commit_info *ci,
@ -1521,6 +1584,8 @@ static enum watch_result our_htlc_spent(struct peer *peer,
log_unusual(peer->log, "Peer redeemed HTLC %zu on-chain using r value", log_unusual(peer->log, "Peer redeemed HTLC %zu on-chain using r value",
i); i);
our_htlc_fulfilled(peer, h, &preimage);
/* BOLT #onchain: /* BOLT #onchain:
* *
* If a node sees a redemption transaction, the output is considered * If a node sees a redemption transaction, the output is considered
@ -1532,6 +1597,36 @@ static enum watch_result our_htlc_spent(struct peer *peer,
return DELETE_WATCH; return DELETE_WATCH;
} }
static void our_htlc_failed(struct peer *peer, struct htlc *htlc)
{
if (htlc->src)
command_htlc_fail(htlc->src->peer, htlc->src);
else
complete_pay_command(peer, htlc);
}
/* We've spent an HTLC output to get our funds back. There's still a
* chance that they could also spend the HTLC output (using the preimage),
* so we need to wait for some confirms.
*
* However, we don't want to wait too long: our upstream will get upset if
* their HTLC has timed out and we don't close it. So we wait one less
* than the HTLC timeout difference.
*/
static enum watch_result our_htlc_timeout_depth(struct peer *peer,
unsigned int depth,
const struct sha256_double *txid,
ptrint_t *i)
{
if (depth == 0)
return KEEP_WATCHING;
if (depth + 1 < peer->dstate->config.min_htlc_expiry)
return KEEP_WATCHING;
our_htlc_failed(peer,
htlc_by_index(peer->closing_onchain.ci, ptr2int(i)));
return DELETE_WATCH;
}
static enum watch_result our_htlc_depth(struct peer *peer, static enum watch_result our_htlc_depth(struct peer *peer,
unsigned int depth, unsigned int depth,
const struct sha256_double *txid, const struct sha256_double *txid,
@ -1574,6 +1669,10 @@ static enum watch_result our_htlc_depth(struct peer *peer,
if (!peer->closing_onchain.resolved[i]) { if (!peer->closing_onchain.resolved[i]) {
peer->closing_onchain.resolved[i] peer->closing_onchain.resolved[i]
= htlc_timeout_tx(peer, peer->closing_onchain.ci, i); = htlc_timeout_tx(peer, peer->closing_onchain.ci, i);
watch_tx(peer->closing_onchain.resolved[i],
peer,
peer->closing_onchain.resolved[i],
our_htlc_timeout_depth, int2ptr(i));
broadcast_tx(peer, peer->closing_onchain.resolved[i]); broadcast_tx(peer, peer->closing_onchain.resolved[i]);
} }
return DELETE_WATCH; return DELETE_WATCH;
@ -1687,15 +1786,36 @@ static void resolve_their_htlcs(struct peer *peer,
size_t i; size_t i;
for (i = start; i < start + num; i++) { for (i = start; i < start + num; i++) {
struct htlc *htlc;
/* Doesn't exist? Resolved by tx itself. */ /* Doesn't exist? Resolved by tx itself. */
if (ci->map[i] == -1) { if (ci->map[i] == -1) {
resolved[i] = tx; resolved[i] = tx;
continue; continue;
} }
/* BOLT #onchain:
*
* If the node ... already knows... a redemption
* preimage for a *commitment tx* output it was offered, it
* MUST *resolve* the output by spending it using the
* preimage.
*/
htlc = htlc_by_index(ci, i);
if (htlc->r) {
peer->closing_onchain.resolved[i]
= htlc_fulfill_tx(peer, ci, i);
broadcast_tx(peer, peer->closing_onchain.resolved[i]);
} else {
/* BOLT #onchain:
*
* Otherwise, if the output HTLC has expired, it is
* considered *irrevocably resolved*.
*/
watch_tx(tx, peer, tx, their_htlc_depth, int2ptr(i)); watch_tx(tx, peer, tx, their_htlc_depth, int2ptr(i));
} }
} }
}
static enum watch_result our_main_output_depth(struct peer *peer, static enum watch_result our_main_output_depth(struct peer *peer,
unsigned int depth, unsigned int depth,
@ -2459,14 +2579,6 @@ free_rest:
tal_free(rest_of_route); tal_free(rest_of_route);
} }
static void our_htlc_failed(struct peer *peer, struct htlc *htlc)
{
if (htlc->src)
command_htlc_fail(htlc->src->peer, htlc->src);
else
complete_pay_command(peer, htlc);
}
/* When changes are committed to. */ /* When changes are committed to. */
void peer_both_committed_to(struct peer *peer, void peer_both_committed_to(struct peer *peer,
const union htlc_staging *changes, const union htlc_staging *changes,

15
daemon/secrets.c

@ -99,6 +99,21 @@ void peer_sign_htlc_refund(const struct peer *peer,
sig); sig);
} }
void peer_sign_htlc_fulfill(const struct peer *peer,
struct bitcoin_tx *spend,
const u8 *htlc_witnessscript,
struct signature *sig)
{
/* Spend tx only has one input: that of the commit tx. */
sign_tx_input(peer->dstate->secpctx,
spend, 0,
NULL, 0,
htlc_witnessscript,
&peer->secrets->final,
&peer->local.finalkey,
sig);
}
void peer_sign_mutual_close(const struct peer *peer, void peer_sign_mutual_close(const struct peer *peer,
struct bitcoin_tx *close, struct bitcoin_tx *close,
struct signature *sig) struct signature *sig)

5
daemon/secrets.h

@ -30,6 +30,11 @@ void peer_sign_htlc_refund(const struct peer *peer,
const u8 *htlc_witnessscript, const u8 *htlc_witnessscript,
struct signature *sig); struct signature *sig);
void peer_sign_htlc_fulfill(const struct peer *peer,
struct bitcoin_tx *spend,
const u8 *htlc_witnessscript,
struct signature *sig);
void peer_sign_mutual_close(const struct peer *peer, void peer_sign_mutual_close(const struct peer *peer,
struct bitcoin_tx *close, struct bitcoin_tx *close,
struct signature *sig); struct signature *sig);

Loading…
Cancel
Save