gossipd: generalize query_channel_range to use a callback.

This means we'll be able to call it for internal reasons, not just dev testing
as now.

Signed-off-by: Rusty Russell <rusty@rustcorp.com.au>
This commit is contained in:
Rusty Russell 2019-09-30 14:27:12 +09:30
parent 18aabc3596
commit c07dff21dc
4 changed files with 85 additions and 37 deletions

View File

@ -631,6 +631,7 @@ static struct io_plan *connectd_new_peer(struct io_conn *conn,
peer->scid_query_nodes_idx = 0; peer->scid_query_nodes_idx = 0;
peer->scid_query_outstanding = false; peer->scid_query_outstanding = false;
peer->query_channel_blocks = NULL; peer->query_channel_blocks = NULL;
peer->query_channel_range_cb = NULL;
peer->num_pings_outstanding = 0; peer->num_pings_outstanding = 0;
peer->gossip_level = peer_gossip_level(daemon, peer->gossip_level = peer_gossip_level(daemon,
peer->gossip_queries_feature); peer->gossip_queries_feature);
@ -1755,7 +1756,7 @@ static struct io_plan *recv_req(struct io_conn *conn,
return send_timestamp_filter(conn, daemon, msg); return send_timestamp_filter(conn, daemon, msg);
case WIRE_GOSSIP_QUERY_CHANNEL_RANGE: case WIRE_GOSSIP_QUERY_CHANNEL_RANGE:
return query_channel_range(conn, daemon, msg); return dev_query_channel_range(conn, daemon, msg);
case WIRE_GOSSIP_DEV_SET_MAX_SCIDS_ENCODE_SIZE: case WIRE_GOSSIP_DEV_SET_MAX_SCIDS_ENCODE_SIZE:
return dev_set_max_scids_encode_size(conn, daemon, msg); return dev_set_max_scids_encode_size(conn, daemon, msg);

View File

@ -114,6 +114,10 @@ struct peer {
u32 range_first_blocknum, range_end_blocknum; u32 range_first_blocknum, range_end_blocknum;
u32 range_blocks_remaining; u32 range_blocks_remaining;
struct short_channel_id *query_channel_scids; struct short_channel_id *query_channel_scids;
void (*query_channel_range_cb)(struct peer *peer,
u32 first_blocknum, u32 number_of_blocks,
const struct short_channel_id *scids,
bool complete);
/* Are we asking this peer to give us lot of gossip? */ /* Are we asking this peer to give us lot of gossip? */
enum gossip_level gossip_level; enum gossip_level gossip_level;

View File

@ -728,17 +728,15 @@ const u8 *handle_reply_channel_range(struct peer *peer, const u8 *msg)
if (peer->range_blocks_remaining) if (peer->range_blocks_remaining)
return NULL; return NULL;
/* All done, send reply to lightningd: that's currently the only thing peer->query_channel_range_cb(peer,
* which triggers this (for testing). Eventually we might start probing first_blocknum,
* for gossip information on our own. */ number_of_blocks,
msg = towire_gossip_query_channel_range_reply(NULL, peer->query_channel_scids,
first_blocknum, complete);
number_of_blocks,
complete,
peer->query_channel_scids);
daemon_conn_send(peer->daemon->master, take(msg));
peer->query_channel_scids = tal_free(peer->query_channel_scids); peer->query_channel_scids = tal_free(peer->query_channel_scids);
peer->query_channel_blocks = tal_free(peer->query_channel_blocks); peer->query_channel_blocks = tal_free(peer->query_channel_blocks);
peer->query_channel_range_cb = NULL;
return NULL; return NULL;
} }
@ -960,6 +958,45 @@ void maybe_send_query_responses(struct peer *peer)
} }
} }
bool query_channel_range(struct daemon *daemon,
struct peer *peer,
u32 first_blocknum, u32 number_of_blocks,
void (*cb)(struct peer *peer,
u32 first_blocknum, u32 number_of_blocks,
const struct short_channel_id *scids,
bool complete))
{
u8 *msg;
assert(peer->gossip_queries_feature);
assert(!peer->query_channel_blocks);
assert(!peer->query_channel_range_cb);
/* Check for overflow on 32-bit machines! */
if (BITMAP_NWORDS(number_of_blocks) < number_of_blocks / BITMAP_WORD_BITS) {
status_broken("query_channel_range: huge number_of_blocks (%u) not supported",
number_of_blocks);
return false;
}
status_debug("sending query_channel_range for blocks %u+%u",
first_blocknum, number_of_blocks);
msg = towire_query_channel_range(NULL, &daemon->chain_hash,
first_blocknum, number_of_blocks,
NULL);
queue_peer_msg(peer, take(msg));
peer->range_first_blocknum = first_blocknum;
peer->range_end_blocknum = first_blocknum + number_of_blocks;
peer->range_blocks_remaining = number_of_blocks;
peer->query_channel_blocks = tal_arrz(peer, bitmap,
BITMAP_NWORDS(number_of_blocks));
peer->query_channel_scids = tal_arr(peer, struct short_channel_id, 0);
peer->query_channel_range_cb = cb;
return true;
}
#if DEVELOPER #if DEVELOPER
struct io_plan *query_scids_req(struct io_conn *conn, struct io_plan *query_scids_req(struct io_conn *conn,
struct daemon *daemon, struct daemon *daemon,
@ -986,11 +1023,23 @@ struct io_plan *query_scids_req(struct io_conn *conn,
return daemon_conn_read_next(conn, daemon->master); return daemon_conn_read_next(conn, daemon->master);
} }
/* FIXME: One day this will be called internally; for now it's just for static void tell_master_scid_range(struct peer *peer,
* testing with dev_query_channel_range. */ u32 first_blocknum, u32 number_of_blocks,
struct io_plan *query_channel_range(struct io_conn *conn, const struct short_channel_id *scids,
struct daemon *daemon, bool complete)
const u8 *msg) {
/* All done, send reply to lightningd. */
u8 *msg = towire_gossip_query_channel_range_reply(NULL,
first_blocknum,
number_of_blocks,
complete,
scids);
daemon_conn_send(peer->daemon->master, take(msg));
}
struct io_plan *dev_query_channel_range(struct io_conn *conn,
struct daemon *daemon,
const u8 *msg)
{ {
struct node_id id; struct node_id id;
u32 first_blocknum, number_of_blocks; u32 first_blocknum, number_of_blocks;
@ -1018,26 +1067,10 @@ struct io_plan *query_channel_range(struct io_conn *conn,
goto fail; goto fail;
} }
/* Check for overflow on 32-bit machines! */ if (!query_channel_range(daemon, peer,
if (BITMAP_NWORDS(number_of_blocks) < number_of_blocks / BITMAP_WORD_BITS) { first_blocknum, number_of_blocks,
status_broken("query_channel_range: huge number_of_blocks (%u) not supported", tell_master_scid_range))
number_of_blocks);
goto fail; goto fail;
}
status_debug("sending query_channel_range for blocks %u+%u",
first_blocknum, number_of_blocks);
msg = towire_query_channel_range(NULL, &daemon->chain_hash,
first_blocknum, number_of_blocks,
NULL);
queue_peer_msg(peer, take(msg));
peer->range_first_blocknum = first_blocknum;
peer->range_end_blocknum = first_blocknum + number_of_blocks;
peer->range_blocks_remaining = number_of_blocks;
peer->query_channel_blocks = tal_arrz(peer, bitmap,
BITMAP_NWORDS(number_of_blocks));
peer->query_channel_scids = tal_arr(peer, struct short_channel_id, 0);
out: out:
return daemon_conn_read_next(conn, daemon->master); return daemon_conn_read_next(conn, daemon->master);

View File

@ -22,14 +22,24 @@ void query_unknown_channel(struct daemon *daemon,
/* This called when the peer is idle. */ /* This called when the peer is idle. */
void maybe_send_query_responses(struct peer *peer); void maybe_send_query_responses(struct peer *peer);
/* Ask this peer for a range of scids. Must support it, and not already
* have a query pending. */
bool query_channel_range(struct daemon *daemon,
struct peer *peer,
u32 first_blocknum, u32 number_of_blocks,
void (*cb)(struct peer *peer,
u32 first_blocknum, u32 number_of_blocks,
const struct short_channel_id *scids,
bool complete));
#if DEVELOPER #if DEVELOPER
struct io_plan *query_scids_req(struct io_conn *conn, struct io_plan *query_scids_req(struct io_conn *conn,
struct daemon *daemon, struct daemon *daemon,
const u8 *msg); const u8 *msg);
struct io_plan *query_channel_range(struct io_conn *conn, struct io_plan *dev_query_channel_range(struct io_conn *conn,
struct daemon *daemon, struct daemon *daemon,
const u8 *msg); const u8 *msg);
/* This is a testing hack to allow us to artificially lower the maximum bytes /* This is a testing hack to allow us to artificially lower the maximum bytes
* of short_channel_ids we'll encode, using dev_set_max_scids_encode_size. */ * of short_channel_ids we'll encode, using dev_set_max_scids_encode_size. */