From fbe6e9e0cfe58afdbbf9dc8bf509aeba9cc7c669 Mon Sep 17 00:00:00 2001 From: Rusty Russell Date: Fri, 25 Sep 2015 11:51:18 +0930 Subject: [PATCH] state: allow multiple SPENDTHEM. Malleability, there could be many of these. Signed-off-by: Rusty Russell --- state.c | 5 ++--- test/test_state_coverage.c | 16 +++++++++++----- 2 files changed, 13 insertions(+), 8 deletions(-) diff --git a/state.c b/state.c index 309b0d2cd..4183dc3ef 100644 --- a/state.c +++ b/state.c @@ -626,9 +626,8 @@ enum state state(const enum state state, const struct state_data *sdata, } /* Now, other side can always spring a commit transaction on us - * (if they haven't already) */ - if (!(bits & STATE_CLOSE_SPENDTHEM_BIT) - && input_is(input, BITCOIN_ANCHOR_THEIRSPEND)) { + * (even if they already have, due to tx malleability) */ + if (input_is(input, BITCOIN_ANCHOR_THEIRSPEND)) { set_effect(effect, broadcast, bitcoin_spend_theirs(effect, sdata)); set_effect(effect, watch, diff --git a/test/test_state_coverage.c b/test/test_state_coverage.c index 416e48178..a48bcc9e0 100644 --- a/test/test_state_coverage.c +++ b/test/test_state_coverage.c @@ -635,12 +635,16 @@ static const char *apply_effects(struct state_data *sdata, sdata->outputs[sdata->num_outputs++] = input_by_name(pkt); } if (effect->watch) { - /* We can have multiple steals in flight, so make an exception - * for BITCOIN_STEAL_DONE */ + /* We can have multiple steals or spendtheirs in flight, + so make exceptions for BITCOIN_STEAL_DONE/BITCOIN_SPEND_THEIRS_DONE */ if (sdata->event_notifies & (1ULL << BITCOIN_STEAL_DONE) & effect->watch->events) effect->watch->events &= ~(1ULL << BITCOIN_STEAL_DONE); + if (sdata->event_notifies & (1ULL << BITCOIN_SPEND_THEIRS_DONE) + & effect->watch->events) + effect->watch->events &= ~(1ULL << BITCOIN_SPEND_THEIRS_DONE); + if (sdata->event_notifies & effect->watch->events) return "event set twice"; sdata->event_notifies |= effect->watch->events; @@ -950,7 +954,7 @@ static struct trail *try_input(const struct state_data *sdata, /* * We expect to loop if: * 1) We deferred, OR - * 2) We get repeated BITCOIN_ANCHOR_OTHERSPEND, OR + * 2) We get repeated BITCOIN_ANCHOR_OTHERSPEND/THEIRSPEND, OR * 3) We pass through NORMAL state. * * And if we're being quick, always stop. @@ -958,6 +962,7 @@ static struct trail *try_input(const struct state_data *sdata, if (effect->defer != INPUT_NONE || newstate == STATE_NORMAL_LOWPRIO || i == BITCOIN_ANCHOR_OTHERSPEND + || i == BITCOIN_ANCHOR_THEIRSPEND || quick) { tal_free(effect); return NULL; @@ -1057,8 +1062,9 @@ static struct trail *run_peer(const struct state_data *sdata, if (!(copy.event_notifies & (1ULL << i))) continue; - /* Don't re-fire (except OTHERSPEND can reoccur) */ - if (i != BITCOIN_ANCHOR_OTHERSPEND) + /* Don't re-fire (except OTHERSPEND/THEIRSPEND can reoccur) */ + if (i != BITCOIN_ANCHOR_OTHERSPEND + && i != BITCOIN_ANCHOR_THEIRSPEND) copy.event_notifies &= ~(1ULL << i); activate_event(©, i); t = try_input(©, i, normalpath, errorpath, depth, hist);