2016-12-21 17:15:35 +01:00
|
|
|
#include "daemon/chaintopology.h"
|
|
|
|
#include "daemon/log.h"
|
|
|
|
#include "daemon/p2p_announce.h"
|
|
|
|
#include "daemon/packets.h"
|
|
|
|
#include "daemon/peer.h"
|
|
|
|
#include "daemon/routing.h"
|
|
|
|
#include "daemon/secrets.h"
|
|
|
|
#include "daemon/timeout.h"
|
2017-01-04 04:39:20 +01:00
|
|
|
#include "utils.h"
|
2016-12-21 17:15:35 +01:00
|
|
|
|
|
|
|
#include <ccan/tal/str/str.h>
|
|
|
|
#include <ccan/tal/tal.h>
|
|
|
|
#include <secp256k1.h>
|
|
|
|
|
2016-12-14 18:59:08 +01:00
|
|
|
struct queued_message {
|
|
|
|
int type;
|
|
|
|
|
|
|
|
/* Unique tag specifying the msg origin */
|
|
|
|
void *tag;
|
|
|
|
|
|
|
|
/* Timestamp for `channel_update`s and `node_announcement`s, 0
|
|
|
|
* for `channel_announcement`s */
|
|
|
|
u32 timestamp;
|
|
|
|
|
|
|
|
/* Serialized payload */
|
|
|
|
u8 *payload;
|
|
|
|
|
|
|
|
struct list_node list;
|
|
|
|
};
|
2016-12-21 17:15:35 +01:00
|
|
|
|
|
|
|
static void broadcast(struct lightningd_state *dstate,
|
|
|
|
int type, u8 *pkt,
|
|
|
|
struct peer *origin)
|
|
|
|
{
|
|
|
|
struct peer *p;
|
|
|
|
list_for_each(&dstate->peers, p, list) {
|
|
|
|
if (state_is_normal(p->state) && origin != p)
|
|
|
|
queue_pkt_nested(p, type, pkt);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-12-14 18:59:08 +01:00
|
|
|
static void queue_broadcast(struct lightningd_state *dstate,
|
|
|
|
const int type,
|
|
|
|
const u32 timestamp,
|
|
|
|
const u8 *tag,
|
2016-12-21 17:07:06 +01:00
|
|
|
const u8 *payload)
|
2016-12-14 18:59:08 +01:00
|
|
|
{
|
|
|
|
struct queued_message *el, *msg;
|
|
|
|
list_for_each(&dstate->broadcast_queue, el, list) {
|
|
|
|
if (el->type == type &&
|
|
|
|
tal_count(tag) == tal_count(el->tag) &&
|
|
|
|
memcmp(el->tag, tag, tal_count(tag)) == 0 &&
|
|
|
|
el->timestamp < timestamp){
|
|
|
|
/* Found a replacement */
|
|
|
|
el->payload = tal_free(el->payload);
|
|
|
|
el->payload = tal_dup_arr(el, u8, payload, tal_count(payload), 0);
|
|
|
|
el->timestamp = timestamp;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* No match found, add a new message to the queue */
|
|
|
|
msg = tal(dstate, struct queued_message);
|
|
|
|
msg->type = type;
|
|
|
|
msg->timestamp = timestamp;
|
|
|
|
msg->tag = tal_dup_arr(msg, u8, tag, tal_count(tag), 0);
|
|
|
|
msg->payload = tal_dup_arr(msg, u8, payload, tal_count(payload), 0);
|
|
|
|
list_add_tail(&dstate->broadcast_queue, &msg->list);
|
|
|
|
}
|
|
|
|
|
2016-12-21 17:15:35 +01:00
|
|
|
void handle_channel_announcement(
|
|
|
|
struct peer *peer,
|
2017-01-04 04:39:20 +01:00
|
|
|
const u8 *announce, size_t len)
|
2016-12-21 17:15:35 +01:00
|
|
|
{
|
|
|
|
u8 *serialized;
|
|
|
|
bool forward = false;
|
2017-01-25 00:33:42 +01:00
|
|
|
secp256k1_ecdsa_signature node_signature_1;
|
|
|
|
secp256k1_ecdsa_signature node_signature_2;
|
2017-01-04 04:39:20 +01:00
|
|
|
struct channel_id channel_id;
|
2017-01-25 00:33:42 +01:00
|
|
|
secp256k1_ecdsa_signature bitcoin_signature_1;
|
|
|
|
secp256k1_ecdsa_signature bitcoin_signature_2;
|
2017-01-04 04:39:20 +01:00
|
|
|
struct pubkey node_id_1;
|
|
|
|
struct pubkey node_id_2;
|
|
|
|
struct pubkey bitcoin_key_1;
|
|
|
|
struct pubkey bitcoin_key_2;
|
|
|
|
const tal_t *tmpctx = tal_tmpctx(peer);
|
2017-01-04 04:39:21 +01:00
|
|
|
u8 *features;
|
2017-01-04 04:39:20 +01:00
|
|
|
|
|
|
|
serialized = tal_dup_arr(tmpctx, u8, announce, len, 0);
|
2017-01-04 04:39:21 +01:00
|
|
|
if (!fromwire_channel_announcement(tmpctx, serialized, NULL,
|
2017-01-04 04:39:20 +01:00
|
|
|
&node_signature_1, &node_signature_2,
|
|
|
|
&bitcoin_signature_1,
|
|
|
|
&bitcoin_signature_2,
|
2017-02-02 05:05:10 +01:00
|
|
|
&channel_id,
|
2017-01-04 04:39:20 +01:00
|
|
|
&node_id_1, &node_id_2,
|
2017-01-04 04:39:21 +01:00
|
|
|
&bitcoin_key_1, &bitcoin_key_2,
|
|
|
|
&features)) {
|
2017-01-04 04:39:20 +01:00
|
|
|
tal_free(tmpctx);
|
2016-12-21 17:15:35 +01:00
|
|
|
return;
|
2017-01-04 04:39:20 +01:00
|
|
|
}
|
2016-12-21 17:15:35 +01:00
|
|
|
|
2017-01-04 04:39:21 +01:00
|
|
|
// FIXME: Check features!
|
2016-12-21 17:15:35 +01:00
|
|
|
//FIXME(cdecker) Check signatures, when the spec is settled
|
|
|
|
//FIXME(cdecker) Check chain topology for the anchor TX
|
|
|
|
|
2016-12-16 22:07:57 +01:00
|
|
|
log_debug(peer->log,
|
|
|
|
"Received channel_announcement for channel %d:%d:%d",
|
2017-01-04 04:39:20 +01:00
|
|
|
channel_id.blocknum,
|
|
|
|
channel_id.txnum,
|
|
|
|
channel_id.outnum
|
2016-12-21 17:15:35 +01:00
|
|
|
);
|
2016-12-16 22:07:57 +01:00
|
|
|
|
2017-01-22 16:16:51 +01:00
|
|
|
forward |= add_channel_direction(peer->dstate->rstate, &node_id_1,
|
2017-01-04 04:39:20 +01:00
|
|
|
&node_id_2, 0, &channel_id,
|
2016-12-16 22:07:57 +01:00
|
|
|
serialized);
|
2017-01-22 16:16:51 +01:00
|
|
|
forward |= add_channel_direction(peer->dstate->rstate, &node_id_2,
|
2017-01-04 04:39:20 +01:00
|
|
|
&node_id_1, 1, &channel_id,
|
2016-12-16 22:07:57 +01:00
|
|
|
serialized);
|
2016-12-21 17:15:35 +01:00
|
|
|
if (!forward){
|
|
|
|
log_debug(peer->log, "Not forwarding channel_announcement");
|
2017-01-04 04:39:20 +01:00
|
|
|
tal_free(tmpctx);
|
2016-12-21 17:15:35 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-01-04 04:39:20 +01:00
|
|
|
u8 *tag = tal_arr(tmpctx, u8, 0);
|
|
|
|
towire_channel_id(&tag, &channel_id);
|
2016-12-16 22:07:57 +01:00
|
|
|
queue_broadcast(peer->dstate, WIRE_CHANNEL_ANNOUNCEMENT,
|
2016-12-14 18:59:08 +01:00
|
|
|
0, /* `channel_announcement`s do not have a timestamp */
|
2016-12-21 17:07:06 +01:00
|
|
|
tag, serialized);
|
2016-12-16 22:07:57 +01:00
|
|
|
|
2017-01-04 04:39:20 +01:00
|
|
|
tal_free(tmpctx);
|
2016-12-21 17:15:35 +01:00
|
|
|
}
|
|
|
|
|
2017-01-04 04:39:20 +01:00
|
|
|
void handle_channel_update(struct peer *peer, const u8 *update, size_t len)
|
2016-12-21 17:15:35 +01:00
|
|
|
{
|
|
|
|
u8 *serialized;
|
|
|
|
struct node_connection *c;
|
2017-01-25 00:33:42 +01:00
|
|
|
secp256k1_ecdsa_signature signature;
|
2017-01-04 04:39:20 +01:00
|
|
|
struct channel_id channel_id;
|
|
|
|
u32 timestamp;
|
|
|
|
u16 flags;
|
|
|
|
u16 expiry;
|
|
|
|
u32 htlc_minimum_msat;
|
|
|
|
u32 fee_base_msat;
|
|
|
|
u32 fee_proportional_millionths;
|
|
|
|
const tal_t *tmpctx = tal_tmpctx(peer);
|
|
|
|
|
|
|
|
serialized = tal_dup_arr(tmpctx, u8, update, len, 0);
|
2017-01-04 04:39:21 +01:00
|
|
|
if (!fromwire_channel_update(serialized, NULL, &signature, &channel_id,
|
2017-01-04 04:39:20 +01:00
|
|
|
×tamp, &flags, &expiry,
|
|
|
|
&htlc_minimum_msat, &fee_base_msat,
|
|
|
|
&fee_proportional_millionths)) {
|
|
|
|
tal_free(tmpctx);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-12-21 17:15:35 +01:00
|
|
|
|
|
|
|
log_debug(peer->log, "Received channel_update for channel %d:%d:%d(%d)",
|
2017-01-04 04:39:20 +01:00
|
|
|
channel_id.blocknum,
|
|
|
|
channel_id.txnum,
|
|
|
|
channel_id.outnum,
|
|
|
|
flags & 0x01
|
2016-12-21 17:15:35 +01:00
|
|
|
);
|
|
|
|
|
2017-01-19 23:46:07 +01:00
|
|
|
c = get_connection_by_cid(peer->dstate->rstate, &channel_id, flags & 0x1);
|
2016-12-21 17:15:35 +01:00
|
|
|
|
|
|
|
if (!c) {
|
|
|
|
log_debug(peer->log, "Ignoring update for unknown channel %d:%d:%d",
|
2017-01-04 04:39:20 +01:00
|
|
|
channel_id.blocknum,
|
|
|
|
channel_id.txnum,
|
|
|
|
channel_id.outnum
|
2016-12-21 17:15:35 +01:00
|
|
|
);
|
2017-01-04 04:39:20 +01:00
|
|
|
tal_free(tmpctx);
|
2016-12-21 17:15:35 +01:00
|
|
|
return;
|
2017-01-04 04:39:20 +01:00
|
|
|
} else if (c->last_timestamp >= timestamp) {
|
2016-12-21 17:15:35 +01:00
|
|
|
log_debug(peer->log, "Ignoring outdated update.");
|
2017-01-04 04:39:20 +01:00
|
|
|
tal_free(tmpctx);
|
2016-12-21 17:15:35 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
//FIXME(cdecker) Check signatures
|
2017-01-04 04:39:20 +01:00
|
|
|
c->last_timestamp = timestamp;
|
|
|
|
c->delay = expiry;
|
|
|
|
c->htlc_minimum_msat = htlc_minimum_msat;
|
|
|
|
c->base_fee = fee_base_msat;
|
|
|
|
c->proportional_fee = fee_proportional_millionths;
|
2016-12-21 17:15:35 +01:00
|
|
|
c->active = true;
|
|
|
|
log_debug(peer->log, "Channel %d:%d:%d(%d) was updated.",
|
2017-01-04 04:39:20 +01:00
|
|
|
channel_id.blocknum,
|
|
|
|
channel_id.txnum,
|
|
|
|
channel_id.outnum,
|
|
|
|
flags
|
2016-12-21 17:15:35 +01:00
|
|
|
);
|
|
|
|
|
2017-01-04 04:39:20 +01:00
|
|
|
u8 *tag = tal_arr(tmpctx, u8, 0);
|
|
|
|
towire_channel_id(&tag, &channel_id);
|
2016-12-14 18:59:08 +01:00
|
|
|
queue_broadcast(peer->dstate,
|
|
|
|
WIRE_CHANNEL_UPDATE,
|
2017-01-04 04:39:20 +01:00
|
|
|
timestamp,
|
2016-12-14 18:59:08 +01:00
|
|
|
tag,
|
2016-12-21 17:07:06 +01:00
|
|
|
serialized);
|
2016-12-16 22:07:57 +01:00
|
|
|
|
|
|
|
tal_free(c->channel_update);
|
2017-01-04 04:39:20 +01:00
|
|
|
c->channel_update = tal_steal(c, serialized);
|
|
|
|
tal_free(tmpctx);
|
2016-12-21 17:15:35 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void handle_node_announcement(
|
2017-01-04 04:39:20 +01:00
|
|
|
struct peer *peer, const u8 *node_ann, size_t len)
|
2016-12-21 17:15:35 +01:00
|
|
|
{
|
|
|
|
u8 *serialized;
|
|
|
|
struct sha256_double hash;
|
|
|
|
struct node *node;
|
2017-01-25 00:33:42 +01:00
|
|
|
secp256k1_ecdsa_signature signature;
|
2017-01-04 04:39:20 +01:00
|
|
|
u32 timestamp;
|
|
|
|
struct pubkey node_id;
|
|
|
|
u8 rgb_color[3];
|
|
|
|
u8 alias[32];
|
2017-01-04 04:39:21 +01:00
|
|
|
u8 *features, *addresses;
|
2017-01-04 04:39:20 +01:00
|
|
|
const tal_t *tmpctx = tal_tmpctx(peer);
|
|
|
|
|
|
|
|
serialized = tal_dup_arr(tmpctx, u8, node_ann, len, 0);
|
2017-01-04 04:39:21 +01:00
|
|
|
if (!fromwire_node_announcement(tmpctx, serialized, NULL,
|
2017-01-04 04:39:21 +01:00
|
|
|
&signature, ×tamp,
|
|
|
|
&node_id, rgb_color, alias, &features,
|
|
|
|
&addresses)) {
|
2017-01-04 04:39:20 +01:00
|
|
|
tal_free(tmpctx);
|
2016-12-21 17:15:35 +01:00
|
|
|
return;
|
2017-01-04 04:39:20 +01:00
|
|
|
}
|
2016-12-21 17:15:35 +01:00
|
|
|
|
2017-01-04 04:39:21 +01:00
|
|
|
// FIXME: Check features!
|
2016-12-21 17:15:35 +01:00
|
|
|
log_debug_struct(peer->log,
|
|
|
|
"Received node_announcement for node %s",
|
2017-01-04 04:39:20 +01:00
|
|
|
struct pubkey, &node_id);
|
2016-12-21 17:15:35 +01:00
|
|
|
|
2017-01-04 04:39:21 +01:00
|
|
|
sha256_double(&hash, serialized + 66, tal_count(serialized) - 66);
|
2017-01-04 04:39:20 +01:00
|
|
|
if (!check_signed_hash(&hash, &signature, &node_id)) {
|
2016-12-21 17:15:35 +01:00
|
|
|
log_debug(peer->dstate->base_log,
|
|
|
|
"Ignoring node announcement, signature verification failed.");
|
2017-01-04 04:39:20 +01:00
|
|
|
tal_free(tmpctx);
|
2016-12-21 17:15:35 +01:00
|
|
|
return;
|
|
|
|
}
|
2017-01-19 23:46:07 +01:00
|
|
|
node = get_node(peer->dstate->rstate, &node_id);
|
2016-12-21 17:15:35 +01:00
|
|
|
|
|
|
|
if (!node) {
|
|
|
|
log_debug(peer->dstate->base_log,
|
|
|
|
"Node not found, was the node_announcement preceeded by at least channel_announcement?");
|
2017-01-04 04:39:20 +01:00
|
|
|
tal_free(tmpctx);
|
2016-12-21 17:15:35 +01:00
|
|
|
return;
|
2017-01-04 04:39:20 +01:00
|
|
|
} else if (node->last_timestamp >= timestamp) {
|
2016-12-21 17:15:35 +01:00
|
|
|
log_debug(peer->dstate->base_log,
|
|
|
|
"Ignoring node announcement, it's outdated.");
|
2017-01-04 04:39:20 +01:00
|
|
|
tal_free(tmpctx);
|
2016-12-21 17:15:35 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-01-04 04:39:20 +01:00
|
|
|
node->last_timestamp = timestamp;
|
2016-12-21 23:49:01 +01:00
|
|
|
node->hostname = tal_free(node->hostname);
|
2017-01-04 04:39:21 +01:00
|
|
|
if (!read_ip(node, addresses, &node->hostname, &node->port)) {
|
|
|
|
/* FIXME: SHOULD fail connection here. */
|
|
|
|
tal_free(serialized);
|
|
|
|
return;
|
|
|
|
}
|
2017-01-04 04:39:20 +01:00
|
|
|
memcpy(node->rgb_color, rgb_color, 3);
|
2016-12-21 17:15:35 +01:00
|
|
|
|
2017-01-04 04:39:20 +01:00
|
|
|
u8 *tag = tal_arr(tmpctx, u8, 0);
|
|
|
|
towire_pubkey(&tag, &node_id);
|
2016-12-14 18:59:08 +01:00
|
|
|
queue_broadcast(peer->dstate,
|
|
|
|
WIRE_NODE_ANNOUNCEMENT,
|
2017-01-04 04:39:20 +01:00
|
|
|
timestamp,
|
2016-12-14 18:59:08 +01:00
|
|
|
tag,
|
2016-12-21 17:07:06 +01:00
|
|
|
serialized);
|
2016-12-16 22:07:57 +01:00
|
|
|
tal_free(node->node_announcement);
|
2017-01-04 04:39:20 +01:00
|
|
|
node->node_announcement = tal_steal(node, serialized);
|
|
|
|
tal_free(tmpctx);
|
2016-12-21 17:15:35 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static void broadcast_channel_update(struct lightningd_state *dstate, struct peer *peer)
|
|
|
|
{
|
|
|
|
struct txlocator *loc;
|
|
|
|
u8 *serialized;
|
2017-01-25 00:33:42 +01:00
|
|
|
secp256k1_ecdsa_signature signature;
|
2017-01-04 04:39:20 +01:00
|
|
|
struct channel_id channel_id;
|
|
|
|
u32 timestamp = time_now().ts.tv_sec;
|
|
|
|
const tal_t *tmpctx = tal_tmpctx(dstate);
|
2016-12-21 17:15:35 +01:00
|
|
|
|
2017-01-04 04:39:20 +01:00
|
|
|
loc = locate_tx(tmpctx, dstate, &peer->anchor.txid);
|
|
|
|
channel_id.blocknum = loc->blkheight;
|
|
|
|
channel_id.txnum = loc->index;
|
|
|
|
channel_id.outnum = peer->anchor.index;
|
2016-12-21 17:15:35 +01:00
|
|
|
|
|
|
|
/* Avoid triggering memcheck */
|
2017-01-04 04:39:20 +01:00
|
|
|
memset(&signature, 0, sizeof(signature));
|
|
|
|
|
|
|
|
serialized = towire_channel_update(tmpctx, &signature, &channel_id,
|
|
|
|
timestamp,
|
|
|
|
pubkey_cmp(&dstate->id, peer->id) > 0,
|
|
|
|
dstate->config.min_htlc_expiry,
|
|
|
|
//FIXME(cdecker) Make the minimum HTLC configurable
|
|
|
|
1,
|
|
|
|
dstate->config.fee_base,
|
|
|
|
dstate->config.fee_per_satoshi);
|
2017-01-04 04:39:21 +01:00
|
|
|
privkey_sign(dstate, serialized + 66, tal_count(serialized) - 66,
|
2017-01-04 04:39:20 +01:00
|
|
|
&signature);
|
|
|
|
serialized = towire_channel_update(tmpctx, &signature, &channel_id,
|
|
|
|
timestamp,
|
|
|
|
pubkey_cmp(&dstate->id, peer->id) > 0,
|
|
|
|
dstate->config.min_htlc_expiry,
|
|
|
|
1,
|
|
|
|
dstate->config.fee_base,
|
|
|
|
dstate->config.fee_per_satoshi);
|
2016-12-21 17:15:35 +01:00
|
|
|
|
|
|
|
broadcast(dstate, WIRE_CHANNEL_UPDATE, serialized, NULL);
|
2017-01-04 04:39:20 +01:00
|
|
|
tal_free(tmpctx);
|
2016-12-21 17:15:35 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static void broadcast_node_announcement(struct lightningd_state *dstate)
|
|
|
|
{
|
|
|
|
u8 *serialized;
|
2017-01-25 00:33:42 +01:00
|
|
|
secp256k1_ecdsa_signature signature;
|
2017-01-04 04:39:20 +01:00
|
|
|
static const u8 rgb_color[3];
|
|
|
|
static const u8 alias[32];
|
|
|
|
u32 timestamp = time_now().ts.tv_sec;
|
|
|
|
const tal_t *tmpctx = tal_tmpctx(dstate);
|
2017-01-04 04:39:21 +01:00
|
|
|
u8 *address;
|
2017-01-04 04:39:20 +01:00
|
|
|
|
|
|
|
/* Are we listening for incoming connections at all? */
|
|
|
|
if (!dstate->external_ip || !dstate->portnum) {
|
|
|
|
tal_free(tmpctx);
|
2016-12-21 17:15:35 +01:00
|
|
|
return;
|
2017-01-04 04:39:20 +01:00
|
|
|
}
|
2016-12-21 17:15:35 +01:00
|
|
|
|
2017-01-04 04:39:20 +01:00
|
|
|
/* Avoid triggering memcheck */
|
|
|
|
memset(&signature, 0, sizeof(signature));
|
|
|
|
|
2017-01-04 04:39:21 +01:00
|
|
|
address = write_ip(tmpctx, dstate->external_ip, dstate->portnum);
|
2017-01-04 04:39:20 +01:00
|
|
|
serialized = towire_node_announcement(tmpctx, &signature,
|
|
|
|
timestamp,
|
2017-01-04 04:39:21 +01:00
|
|
|
&dstate->id, rgb_color, alias,
|
2017-01-10 05:49:25 +01:00
|
|
|
NULL,
|
|
|
|
address);
|
2017-01-04 04:39:21 +01:00
|
|
|
privkey_sign(dstate, serialized + 66, tal_count(serialized) - 66,
|
2017-01-04 04:39:20 +01:00
|
|
|
&signature);
|
|
|
|
serialized = towire_node_announcement(tmpctx, &signature,
|
|
|
|
timestamp,
|
2017-01-04 04:39:21 +01:00
|
|
|
&dstate->id, rgb_color, alias,
|
2017-01-10 05:49:25 +01:00
|
|
|
NULL,
|
|
|
|
address);
|
2016-12-21 17:15:35 +01:00
|
|
|
broadcast(dstate, WIRE_NODE_ANNOUNCEMENT, serialized, NULL);
|
2017-01-04 04:39:20 +01:00
|
|
|
tal_free(tmpctx);
|
2016-12-21 17:15:35 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static void broadcast_channel_announcement(struct lightningd_state *dstate, struct peer *peer)
|
|
|
|
{
|
|
|
|
struct txlocator *loc;
|
2017-01-04 04:39:20 +01:00
|
|
|
struct channel_id channel_id;
|
2017-01-25 00:33:42 +01:00
|
|
|
secp256k1_ecdsa_signature node_signature[2];
|
|
|
|
secp256k1_ecdsa_signature bitcoin_signature[2];
|
2017-01-04 04:39:20 +01:00
|
|
|
const struct pubkey *node_id[2];
|
|
|
|
const struct pubkey *bitcoin_key[2];
|
2017-01-25 00:33:42 +01:00
|
|
|
secp256k1_ecdsa_signature *my_node_signature;
|
|
|
|
secp256k1_ecdsa_signature *my_bitcoin_signature;
|
2016-12-13 22:59:40 +01:00
|
|
|
u8 *serialized;
|
2017-01-04 04:39:20 +01:00
|
|
|
const tal_t *tmpctx = tal_tmpctx(dstate);
|
2016-12-21 17:15:35 +01:00
|
|
|
|
2017-01-04 04:39:20 +01:00
|
|
|
loc = locate_tx(tmpctx, dstate, &peer->anchor.txid);
|
2016-12-21 17:15:35 +01:00
|
|
|
|
2017-01-04 04:39:20 +01:00
|
|
|
channel_id.blocknum = loc->blkheight;
|
|
|
|
channel_id.txnum = loc->index;
|
|
|
|
channel_id.outnum = peer->anchor.index;
|
2016-12-13 22:59:40 +01:00
|
|
|
|
|
|
|
/* Set all sigs to zero */
|
2017-01-04 04:39:20 +01:00
|
|
|
memset(node_signature, 0, sizeof(node_signature));
|
|
|
|
memset(bitcoin_signature, 0, sizeof(bitcoin_signature));
|
2016-12-13 22:59:40 +01:00
|
|
|
|
|
|
|
//FIXME(cdecker) Copy remote stored signatures into place
|
2016-12-21 17:15:35 +01:00
|
|
|
if (pubkey_cmp(&dstate->id, peer->id) > 0) {
|
2017-01-04 04:39:20 +01:00
|
|
|
node_id[0] = peer->id;
|
|
|
|
node_id[1] = &dstate->id;
|
|
|
|
bitcoin_key[0] = peer->id;
|
|
|
|
bitcoin_key[1] = &dstate->id;
|
|
|
|
my_node_signature = &node_signature[1];
|
|
|
|
my_bitcoin_signature = &bitcoin_signature[1];
|
2016-12-21 17:15:35 +01:00
|
|
|
} else {
|
2017-01-04 04:39:20 +01:00
|
|
|
node_id[1] = peer->id;
|
|
|
|
node_id[0] = &dstate->id;
|
|
|
|
bitcoin_key[1] = peer->id;
|
|
|
|
bitcoin_key[0] = &dstate->id;
|
|
|
|
my_node_signature = &node_signature[0];
|
|
|
|
my_bitcoin_signature = &bitcoin_signature[0];
|
2016-12-21 17:15:35 +01:00
|
|
|
}
|
2017-01-04 04:39:20 +01:00
|
|
|
|
2016-12-13 22:59:40 +01:00
|
|
|
/* Sign the node_id with the bitcoin_key, proves delegation */
|
2017-01-04 04:39:20 +01:00
|
|
|
serialized = tal_arr(tmpctx, u8, 0);
|
2016-12-13 22:59:40 +01:00
|
|
|
towire_pubkey(&serialized, &dstate->id);
|
|
|
|
privkey_sign(dstate, serialized, tal_count(serialized), my_bitcoin_signature);
|
2016-12-21 17:15:35 +01:00
|
|
|
|
2017-02-02 05:05:10 +01:00
|
|
|
/* BOLT #7:
|
|
|
|
*
|
|
|
|
* The creating node MUST compute the double-SHA256 hash `h` of the
|
|
|
|
* message, starting at offset 256, up to the end of the message.
|
|
|
|
*/
|
2017-01-04 04:39:20 +01:00
|
|
|
serialized = towire_channel_announcement(tmpctx, &node_signature[0],
|
|
|
|
&node_signature[1],
|
|
|
|
&bitcoin_signature[0],
|
|
|
|
&bitcoin_signature[1],
|
2017-02-02 05:05:10 +01:00
|
|
|
&channel_id,
|
2017-01-04 04:39:20 +01:00
|
|
|
node_id[0],
|
|
|
|
node_id[1],
|
|
|
|
bitcoin_key[0],
|
2017-01-04 04:39:21 +01:00
|
|
|
bitcoin_key[1],
|
2017-01-10 05:49:25 +01:00
|
|
|
NULL);
|
2017-02-02 05:05:10 +01:00
|
|
|
privkey_sign(dstate, serialized + 256, tal_count(serialized) - 256, my_node_signature);
|
2016-12-13 22:59:40 +01:00
|
|
|
|
2017-01-04 04:39:20 +01:00
|
|
|
serialized = towire_channel_announcement(tmpctx, &node_signature[0],
|
|
|
|
&node_signature[1],
|
|
|
|
&bitcoin_signature[0],
|
|
|
|
&bitcoin_signature[1],
|
2017-02-02 05:05:10 +01:00
|
|
|
&channel_id,
|
2017-01-04 04:39:20 +01:00
|
|
|
node_id[0],
|
|
|
|
node_id[1],
|
|
|
|
bitcoin_key[0],
|
2017-01-04 04:39:21 +01:00
|
|
|
bitcoin_key[1],
|
2017-01-10 05:49:25 +01:00
|
|
|
NULL);
|
2016-12-13 22:59:40 +01:00
|
|
|
broadcast(dstate, WIRE_CHANNEL_ANNOUNCEMENT, serialized, NULL);
|
2017-01-04 04:39:20 +01:00
|
|
|
tal_free(tmpctx);
|
2016-12-21 17:15:35 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static void announce(struct lightningd_state *dstate)
|
|
|
|
{
|
|
|
|
struct peer *p;
|
|
|
|
int nchan = 0;
|
|
|
|
|
2016-12-15 16:44:29 +01:00
|
|
|
new_reltimer(dstate, dstate, time_from_sec(5*60*60), announce, dstate);
|
2016-12-21 17:15:35 +01:00
|
|
|
|
|
|
|
list_for_each(&dstate->peers, p, list) {
|
|
|
|
if (state_is_normal(p->state)) {
|
|
|
|
broadcast_channel_announcement(dstate, p);
|
|
|
|
broadcast_channel_update(dstate, p);
|
|
|
|
nchan += 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* No point in broadcasting our node if we don't have a channel */
|
|
|
|
if (nchan > 0)
|
|
|
|
broadcast_node_announcement(dstate);
|
|
|
|
}
|
|
|
|
|
|
|
|
void announce_channel(struct lightningd_state *dstate, struct peer *peer)
|
|
|
|
{
|
|
|
|
broadcast_channel_announcement(dstate, peer);
|
|
|
|
broadcast_channel_update(dstate, peer);
|
|
|
|
broadcast_node_announcement(dstate);
|
2016-12-14 18:59:08 +01:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
static void process_broadcast_queue(struct lightningd_state *dstate)
|
|
|
|
{
|
|
|
|
new_reltimer(dstate, dstate, time_from_sec(30), process_broadcast_queue, dstate);
|
|
|
|
struct queued_message *el;
|
|
|
|
while ((el = list_pop(&dstate->broadcast_queue, struct queued_message, list)) != NULL) {
|
|
|
|
broadcast(dstate, el->type, el->payload, NULL);
|
|
|
|
tal_free(el);
|
|
|
|
}
|
2016-12-21 17:15:35 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void setup_p2p_announce(struct lightningd_state *dstate)
|
|
|
|
{
|
2016-12-15 16:44:29 +01:00
|
|
|
new_reltimer(dstate, dstate, time_from_sec(5*60*60), announce, dstate);
|
2016-12-14 18:59:08 +01:00
|
|
|
new_reltimer(dstate, dstate, time_from_sec(30), process_broadcast_queue, dstate);
|
2016-12-21 17:15:35 +01:00
|
|
|
}
|