From 2dba51f4553099e97606f20d3a874cd40eb16360 Mon Sep 17 00:00:00 2001 From: niftynei Date: Thu, 2 Feb 2023 17:32:35 -0600 Subject: [PATCH 1/4] rfc-dual-fund: update to latest spec for dual-funding - Renamed zerod_channel_ids to temporary_channel_id - Renamed witness_stack->witnesses - Renamed witness_element->witness_elements - open_channel2 now includes second commitment point - accept_channel2 now includes second commitment point Current commit on rfc branch 64f7f360b9f3c2664d078e2129cfe83098fc4617 Changelog-EXPERIMENTAL: Protocol: dual-funding spec changed in incompatible ways, won't work with old versions (but maybe soon with Eclair!!) --- common/psbt_internal.c | 12 ++-- openingd/dualopend.c | 9 ++- ...racted_peer_07_openchannelv2_updates.patch | 59 +++++++++++++++++++ wire/peer_wire.csv | 16 ++--- 4 files changed, 81 insertions(+), 15 deletions(-) create mode 100644 wire/extracted_peer_07_openchannelv2_updates.patch diff --git a/common/psbt_internal.c b/common/psbt_internal.c index 46bb0d134fd2..bc7ca0a9c7ca 100644 --- a/common/psbt_internal.c +++ b/common/psbt_internal.c @@ -17,8 +17,8 @@ psbt_input_set_final_witness_stack(const tal_t *ctx, for (size_t i = 0; i < tal_count(elements); i++) wally_tx_witness_stack_add(in->final_witness, - elements[i]->witness, - tal_bytelen(elements[i]->witness)); + elements[i]->witness_data, + tal_bytelen(elements[i]->witness_data)); tal_wally_end(ctx); } @@ -78,13 +78,13 @@ psbt_to_witness_stacks(const tal_t *ctx, tal(stacks, struct witness_stack); /* Convert the wally_tx_witness_stack to * a witness_stack entry */ - stack->witness_element = + stack->witness_elements = tal_arr(stack, struct witness_element *, wtx_s->num_items); - for (size_t j = 0; j < tal_count(stack->witness_element); j++) { - stack->witness_element[j] = tal(stack, + for (size_t j = 0; j < tal_count(stack->witness_elements); j++) { + stack->witness_elements[j] = tal(stack, struct witness_element); - stack->witness_element[j]->witness = + stack->witness_elements[j]->witness_data = tal_dup_arr(stack, u8, wtx_s->items[j].witness, wtx_s->items[j].witness_len, diff --git a/openingd/dualopend.c b/openingd/dualopend.c index 4b2956f97508..963806338017 100644 --- a/openingd/dualopend.c +++ b/openingd/dualopend.c @@ -171,9 +171,10 @@ struct state { /* Information we need between funding_start and funding_complete */ struct basepoints their_points; - /* hsmd gives us our first per-commitment point, and peer tells us + /* hsmd gives us our first+second per-commitment points, and peer tells us * theirs */ struct pubkey first_per_commitment_point[NUM_SIDES]; + struct pubkey second_per_commitment_point[NUM_SIDES]; struct channel_id channel_id; u8 channel_flags; @@ -1101,7 +1102,7 @@ static void handle_tx_sigs(struct state *state, const u8 *msg) tal_hex(msg, msg)); elem = cast_const2(const struct witness_element **, - ws[j++]->witness_element); + ws[j++]->witness_elements); psbt_finalize_input(tx_state->psbt, in, elem); } @@ -2231,6 +2232,7 @@ static void accepter_start(struct state *state, const u8 *oc2_msg) &state->their_points.delayed_payment, &state->their_points.htlc, &state->first_per_commitment_point[REMOTE], + &state->second_per_commitment_point[REMOTE], &state->channel_flags, &open_tlv)) open_err_fatal(state, "Parsing open_channel2 %s", @@ -2549,6 +2551,7 @@ static void accepter_start(struct state *state, const u8 *oc2_msg) &state->our_points.delayed_payment, &state->our_points.htlc, &state->first_per_commitment_point[LOCAL], + &state->second_per_commitment_point[LOCAL], a_tlv); /* Everything's ok. Let's figure out the actual channel_id now */ @@ -3004,6 +3007,7 @@ static void opener_start(struct state *state, u8 *msg) &state->our_points.delayed_payment, &state->our_points.htlc, &state->first_per_commitment_point[LOCAL], + &state->second_per_commitment_point[LOCAL], state->channel_flags, open_tlv); @@ -3032,6 +3036,7 @@ static void opener_start(struct state *state, u8 *msg) &state->their_points.delayed_payment, &state->their_points.htlc, &state->first_per_commitment_point[REMOTE], + &state->second_per_commitment_point[REMOTE], &a_tlv)) open_err_fatal(state, "Parsing accept_channel2 %s", tal_hex(msg, msg)); diff --git a/wire/extracted_peer_07_openchannelv2_updates.patch b/wire/extracted_peer_07_openchannelv2_updates.patch new file mode 100644 index 000000000000..05a281ac5708 --- /dev/null +++ b/wire/extracted_peer_07_openchannelv2_updates.patch @@ -0,0 +1,59 @@ +--- wire/peer_wire.csv 2023-02-02 17:51:50.435463786 -0600 ++++ - 2023-02-02 17:51:56.693837258 -0600 +@@ -62,13 +63,13 @@ + msgdata,tx_signatures,channel_id,channel_id, + msgdata,tx_signatures,txid,sha256, + msgdata,tx_signatures,num_witnesses,u16, +-msgdata,tx_signatures,witness_stack,witness_stack,num_witnesses ++msgdata,tx_signatures,witnesses,witness_stack,num_witnesses + subtype,witness_stack +-subtypedata,witness_stack,num_input_witness,u16, +-subtypedata,witness_stack,witness_element,witness_element,num_input_witness ++subtypedata,witness_stack,num_witness_elements,u16, ++subtypedata,witness_stack,witness_elements,witness_element,num_witness_elements + subtype,witness_element + subtypedata,witness_element,len,u16, +-subtypedata,witness_element,witness,byte,len ++subtypedata,witness_element,witness_data,byte,len + msgtype,tx_init_rbf,72 + msgdata,tx_init_rbf,channel_id,channel_id, + msgdata,tx_init_rbf,locktime,u32, +@@ -145,7 +146,7 @@ + tlvdata,channel_ready_tlvs,short_channel_id,alias,short_channel_id, + msgtype,open_channel2,64 + msgdata,open_channel2,chain_hash,chain_hash, +-msgdata,open_channel2,zerod_channel_id,channel_id, ++msgdata,open_channel2,temporary_channel_id,channel_id, + msgdata,open_channel2,funding_feerate_perkw,u32, + msgdata,open_channel2,commitment_feerate_perkw,u32, + msgdata,open_channel2,funding_satoshis,u64, +@@ -161,19 +162,20 @@ + msgdata,open_channel2,delayed_payment_basepoint,point, + msgdata,open_channel2,htlc_basepoint,point, + msgdata,open_channel2,first_per_commitment_point,point, ++msgdata,open_channel2,second_per_commitment_point,point, + msgdata,open_channel2,channel_flags,byte, + msgdata,open_channel2,tlvs,opening_tlvs, + tlvtype,opening_tlvs,upfront_shutdown_script,0 + tlvdata,opening_tlvs,upfront_shutdown_script,shutdown_scriptpubkey,byte,... + tlvtype,opening_tlvs,channel_type,1 + tlvdata,opening_tlvs,channel_type,type,byte,... + tlvtype,opening_tlvs,request_funds,3 + tlvdata,opening_tlvs,request_funds,requested_sats,u64, + tlvdata,opening_tlvs,request_funds,blockheight,u32, + tlvtype,opening_tlvs,require_confirmed_inputs,2 + tlvdata,opening_tlvs,require_confirmed_inputs,empty,byte,0 + msgtype,accept_channel2,65 +-msgdata,accept_channel2,zerod_channel_id,channel_id, ++msgdata,accept_channel2,temporary_channel_id,channel_id, + msgdata,accept_channel2,funding_satoshis,u64, + msgdata,accept_channel2,dust_limit_satoshis,u64, + msgdata,accept_channel2,max_htlc_value_in_flight_msat,u64, +@@ -187,6 +186,7 @@ + msgdata,accept_channel2,delayed_payment_basepoint,point, + msgdata,accept_channel2,htlc_basepoint,point, + msgdata,accept_channel2,first_per_commitment_point,point, ++msgdata,accept_channel2,second_per_commitment_point,point, + msgdata,accept_channel2,tlvs,accept_tlvs, + tlvtype,accept_tlvs,upfront_shutdown_script,0 + tlvdata,accept_tlvs,upfront_shutdown_script,shutdown_scriptpubkey,byte,... diff --git a/wire/peer_wire.csv b/wire/peer_wire.csv index 398a7b4352bd..b01d1f40dc4b 100644 --- a/wire/peer_wire.csv +++ b/wire/peer_wire.csv @@ -62,13 +62,13 @@ msgtype,tx_signatures,71 msgdata,tx_signatures,channel_id,channel_id, msgdata,tx_signatures,txid,sha256, msgdata,tx_signatures,num_witnesses,u16, -msgdata,tx_signatures,witness_stack,witness_stack,num_witnesses +msgdata,tx_signatures,witnesses,witness_stack,num_witnesses subtype,witness_stack -subtypedata,witness_stack,num_input_witness,u16, -subtypedata,witness_stack,witness_element,witness_element,num_input_witness +subtypedata,witness_stack,num_witness_elements,u16, +subtypedata,witness_stack,witness_elements,witness_element,num_witness_elements subtype,witness_element subtypedata,witness_element,len,u16, -subtypedata,witness_element,witness,byte,len +subtypedata,witness_element,witness_data,byte,len msgtype,tx_init_rbf,72 msgdata,tx_init_rbf,channel_id,channel_id, msgdata,tx_init_rbf,locktime,u32, @@ -145,7 +145,7 @@ tlvtype,channel_ready_tlvs,short_channel_id,1 tlvdata,channel_ready_tlvs,short_channel_id,alias,short_channel_id, msgtype,open_channel2,64 msgdata,open_channel2,chain_hash,chain_hash, -msgdata,open_channel2,zerod_channel_id,channel_id, +msgdata,open_channel2,temporary_channel_id,channel_id, msgdata,open_channel2,funding_feerate_perkw,u32, msgdata,open_channel2,commitment_feerate_perkw,u32, msgdata,open_channel2,funding_satoshis,u64, @@ -161,6 +161,7 @@ msgdata,open_channel2,payment_basepoint,point, msgdata,open_channel2,delayed_payment_basepoint,point, msgdata,open_channel2,htlc_basepoint,point, msgdata,open_channel2,first_per_commitment_point,point, +msgdata,open_channel2,second_per_commitment_point,point, msgdata,open_channel2,channel_flags,byte, msgdata,open_channel2,tlvs,opening_tlvs, tlvtype,opening_tlvs,upfront_shutdown_script,0 @@ -172,7 +173,7 @@ tlvdata,opening_tlvs,request_funds,requested_sats,u64, tlvdata,opening_tlvs,request_funds,blockheight,u32, tlvtype,opening_tlvs,require_confirmed_inputs,2 msgtype,accept_channel2,65 -msgdata,accept_channel2,zerod_channel_id,channel_id, +msgdata,accept_channel2,temporary_channel_id,channel_id, msgdata,accept_channel2,funding_satoshis,u64, msgdata,accept_channel2,dust_limit_satoshis,u64, msgdata,accept_channel2,max_htlc_value_in_flight_msat,u64, @@ -186,6 +187,7 @@ msgdata,accept_channel2,payment_basepoint,point, msgdata,accept_channel2,delayed_payment_basepoint,point, msgdata,accept_channel2,htlc_basepoint,point, msgdata,accept_channel2,first_per_commitment_point,point, +msgdata,accept_channel2,second_per_commitment_point,point, msgdata,accept_channel2,tlvs,accept_tlvs, tlvtype,accept_tlvs,upfront_shutdown_script,0 tlvdata,accept_tlvs,upfront_shutdown_script,shutdown_scriptpubkey,byte,... @@ -225,7 +227,7 @@ msgdata,update_add_htlc,payment_hash,sha256, msgdata,update_add_htlc,cltv_expiry,u32, msgdata,update_add_htlc,onion_routing_packet,byte,1366 msgdata,update_add_htlc,tlvs,update_add_tlvs, -tlvtype,update_add_tlvs,blinding,2 +tlvtype,update_add_tlvs,blinding,0 tlvdata,update_add_tlvs,blinding,blinding,point, msgtype,update_fulfill_htlc,130 msgdata,update_fulfill_htlc,channel_id,channel_id, From 3833aa544f03b35a9e3b4fd2aad263c29c100b62 Mon Sep 17 00:00:00 2001 From: niftynei Date: Thu, 2 Feb 2023 17:55:48 -0600 Subject: [PATCH 2/4] df: fetch both the first+second commitment point Ignore the sent back second commitment point that we get; we'll get it again at `channel_ready`. --- openingd/dualopend.c | 40 +++++++++++++++++++++++++++------------- 1 file changed, 27 insertions(+), 13 deletions(-) diff --git a/openingd/dualopend.c b/openingd/dualopend.c index 963806338017..e7da64ac922f 100644 --- a/openingd/dualopend.c +++ b/openingd/dualopend.c @@ -2232,6 +2232,8 @@ static void accepter_start(struct state *state, const u8 *oc2_msg) &state->their_points.delayed_payment, &state->their_points.htlc, &state->first_per_commitment_point[REMOTE], + /* We don't actually do anything with this currently, + * as they send it to us again in `channel_ready` */ &state->second_per_commitment_point[REMOTE], &state->channel_flags, &open_tlv)) @@ -3036,6 +3038,8 @@ static void opener_start(struct state *state, u8 *msg) &state->their_points.delayed_payment, &state->their_points.htlc, &state->first_per_commitment_point[REMOTE], + /* We don't actually do anything with this currently, + * as they send it to us again in `channel_ready` */ &state->second_per_commitment_point[REMOTE], &a_tlv)) open_err_fatal(state, "Parsing accept_channel2 %s", @@ -4197,13 +4201,33 @@ static u8 *handle_peer_in(struct state *state) peer_failed_connection_lost(); } +static void fetch_per_commitment_point(u32 point_count, + struct pubkey *commit_point) +{ + u8 *msg; + struct secret *none; + + wire_sync_write(HSM_FD, + take(towire_hsmd_get_per_commitment_point(NULL, point_count))); + msg = wire_sync_read(tmpctx, HSM_FD); + if (!fromwire_hsmd_get_per_commitment_point_reply(tmpctx, msg, + commit_point, + &none)) + status_failed(STATUS_FAIL_HSM_IO, + "Bad get_per_commitment_point_reply %s", + tal_hex(tmpctx, msg)); + + /*~ The HSM gives us the N-2'th per-commitment secret when we get the + * N'th per-commitment point. But since N=0, it won't give us one. */ + assert(none == NULL); +} + int main(int argc, char *argv[]) { common_setup(argv[0]); struct pollfd pollfd[2]; struct state *state = tal(NULL, struct state); - struct secret *none; struct fee_states *fee_states; enum side opener; u8 *msg; @@ -4362,18 +4386,8 @@ int main(int argc, char *argv[]) /*~ We need an initial per-commitment point whether we're funding or * they are, and lightningd has reserved a unique dbid for us already, * so we might as well get the hsm daemon to generate it now. */ - wire_sync_write(HSM_FD, - take(towire_hsmd_get_per_commitment_point(NULL, 0))); - msg = wire_sync_read(tmpctx, HSM_FD); - if (!fromwire_hsmd_get_per_commitment_point_reply(tmpctx, msg, - &state->first_per_commitment_point[LOCAL], - &none)) - status_failed(STATUS_FAIL_HSM_IO, - "Bad get_per_commitment_point_reply %s", - tal_hex(tmpctx, msg)); - /*~ The HSM gives us the N-2'th per-commitment secret when we get the - * N'th per-commitment point. But since N=0, it won't give us one. */ - assert(none == NULL); + fetch_per_commitment_point(0, &state->first_per_commitment_point[LOCAL]); + fetch_per_commitment_point(1, &state->second_per_commitment_point[LOCAL]); /*~ We manually run a little poll() loop here. With only two fds */ pollfd[0].fd = REQ_FD; From 6bb55bc6ce81b2d7057e54e50ea6ea70e7490b20 Mon Sep 17 00:00:00 2001 From: niftynei Date: Mon, 6 Feb 2023 16:22:23 -0600 Subject: [PATCH 3/4] df: remove minimum witness weight for input calculations We can't know how much taproot etc inputs weight will be, so we just make sure that a peer covers the known bytes, at least. --- openingd/dualopend.c | 5 ----- 1 file changed, 5 deletions(-) diff --git a/openingd/dualopend.c b/openingd/dualopend.c index e7da64ac922f..f213f2c02b7f 100644 --- a/openingd/dualopend.c +++ b/openingd/dualopend.c @@ -603,11 +603,6 @@ static size_t psbt_input_weight(struct wally_psbt *psbt, (psbt->inputs[in].redeem_script_len + (varint_t) varint_size(psbt->inputs[in].redeem_script_len)) * 4; - /* BOLT-f53ca2301232db780843e894f55d95d512f297f9 #3: - * - * The minimum witness weight for an input is 110. - */ - weight += 110; return weight; } From c7d8c297f6c9a6358372d456a5ab4ec1dbcf178c Mon Sep 17 00:00:00 2001 From: niftynei Date: Tue, 7 Feb 2023 13:49:53 -0600 Subject: [PATCH 4/4] df: remove static remote key dependency Must be negotiated independently. Requested-By: @t-bast --- common/features.c | 6 ------ 1 file changed, 6 deletions(-) diff --git a/common/features.c b/common/features.c index 203c0d2f39b8..de3e98fd6c28 100644 --- a/common/features.c +++ b/common/features.c @@ -168,12 +168,6 @@ static const struct dependency feature_deps[] = { * `option_anchors_zero_fee_htlc_tx` | ... | ... | `option_static_remotekey` */ { OPT_ANCHORS_ZERO_FEE_HTLC_TX, OPT_STATIC_REMOTEKEY }, - /* BOLT-f53ca2301232db780843e894f55d95d512f297f9 #9: - * Name | Description | Context | Dependencies | - * ... - * `option_dual_fund` | ... | ... | `option_static_remotekey` - */ - { OPT_DUAL_FUND, OPT_STATIC_REMOTEKEY }, /* BOLT-route-blinding #9: * Name | Description | Context | Dependencies | * ...