diff --git a/gossipd/gossipd.c b/gossipd/gossipd.c index 1990e7cfa..5161f4d54 100644 --- a/gossipd/gossipd.c +++ b/gossipd/gossipd.c @@ -631,6 +631,7 @@ static struct io_plan *connectd_new_peer(struct io_conn *conn, peer->scid_query_nodes_idx = 0; peer->scid_query_outstanding = false; peer->query_channel_blocks = NULL; + peer->query_channel_range_cb = NULL; peer->num_pings_outstanding = 0; peer->gossip_level = peer_gossip_level(daemon, 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); 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: return dev_set_max_scids_encode_size(conn, daemon, msg); diff --git a/gossipd/gossipd.h b/gossipd/gossipd.h index d5a23ef8a..0663bbfa6 100644 --- a/gossipd/gossipd.h +++ b/gossipd/gossipd.h @@ -114,6 +114,10 @@ struct peer { u32 range_first_blocknum, range_end_blocknum; u32 range_blocks_remaining; 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? */ enum gossip_level gossip_level; diff --git a/gossipd/queries.c b/gossipd/queries.c index 7affba320..f6d076dfa 100644 --- a/gossipd/queries.c +++ b/gossipd/queries.c @@ -728,17 +728,15 @@ const u8 *handle_reply_channel_range(struct peer *peer, const u8 *msg) if (peer->range_blocks_remaining) return NULL; - /* All done, send reply to lightningd: that's currently the only thing - * which triggers this (for testing). Eventually we might start probing - * for gossip information on our own. */ - msg = towire_gossip_query_channel_range_reply(NULL, - first_blocknum, - number_of_blocks, - complete, - peer->query_channel_scids); - daemon_conn_send(peer->daemon->master, take(msg)); + peer->query_channel_range_cb(peer, + first_blocknum, + number_of_blocks, + peer->query_channel_scids, + complete); + peer->query_channel_scids = tal_free(peer->query_channel_scids); peer->query_channel_blocks = tal_free(peer->query_channel_blocks); + peer->query_channel_range_cb = 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 struct io_plan *query_scids_req(struct io_conn *conn, 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); } -/* FIXME: One day this will be called internally; for now it's just for - * testing with dev_query_channel_range. */ -struct io_plan *query_channel_range(struct io_conn *conn, - struct daemon *daemon, - const u8 *msg) +static void tell_master_scid_range(struct peer *peer, + u32 first_blocknum, u32 number_of_blocks, + const struct short_channel_id *scids, + bool complete) +{ + /* 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; u32 first_blocknum, number_of_blocks; @@ -1018,26 +1067,10 @@ struct io_plan *query_channel_range(struct io_conn *conn, goto fail; } - /* 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); + if (!query_channel_range(daemon, peer, + first_blocknum, number_of_blocks, + tell_master_scid_range)) 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: return daemon_conn_read_next(conn, daemon->master); diff --git a/gossipd/queries.h b/gossipd/queries.h index 19c271109..7f442df6d 100644 --- a/gossipd/queries.h +++ b/gossipd/queries.h @@ -22,14 +22,24 @@ void query_unknown_channel(struct daemon *daemon, /* This called when the peer is idle. */ 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 struct io_plan *query_scids_req(struct io_conn *conn, struct daemon *daemon, const u8 *msg); -struct io_plan *query_channel_range(struct io_conn *conn, - struct daemon *daemon, - const u8 *msg); +struct io_plan *dev_query_channel_range(struct io_conn *conn, + struct daemon *daemon, + const u8 *msg); /* 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. */