seeker: try asking peer which gave us unknown data first.

This should give more reliable results, though it risks us getting
suckered into always consulting the same peer.

Signed-off-by: Rusty Russell <rusty@rustcorp.com.au>
This commit is contained in:
Rusty Russell 2019-10-08 12:01:24 +10:30 committed by neil saitug
parent 82a5efa932
commit f7cffbad98
2 changed files with 51 additions and 13 deletions

View File

@ -98,6 +98,10 @@ struct seeker {
/* Peers we've asked to stream us gossip */ /* Peers we've asked to stream us gossip */
struct peer *gossiper_softref[3]; struct peer *gossiper_softref[3];
/* A peer that told us about unknown gossip. */
struct peer *preferred_peer_softref;
}; };
/* Mutual recursion */ /* Mutual recursion */
@ -143,6 +147,7 @@ struct seeker *new_seeker(struct daemon *daemon)
seeker->random_peer_softref = NULL; seeker->random_peer_softref = NULL;
for (size_t i = 0; i < ARRAY_SIZE(seeker->gossiper_softref); i++) for (size_t i = 0; i < ARRAY_SIZE(seeker->gossiper_softref); i++)
seeker->gossiper_softref[i] = NULL; seeker->gossiper_softref[i] = NULL;
seeker->preferred_peer_softref = NULL;
set_state(seeker, STARTING_UP); set_state(seeker, STARTING_UP);
begin_check_timer(seeker); begin_check_timer(seeker);
memleak_add_helper(seeker, memleak_help_seeker); memleak_add_helper(seeker, memleak_help_seeker);
@ -165,6 +170,30 @@ static bool selected_peer(struct seeker *seeker, struct peer *peer)
return true; return true;
} }
static void set_preferred_peer(struct seeker *seeker, struct peer *peer)
{
if (seeker->preferred_peer_softref
&& seeker->preferred_peer_softref != peer) {
clear_softref(seeker, &seeker->preferred_peer_softref);
set_softref(seeker, &seeker->preferred_peer_softref, peer);
}
}
/* Get a random peer, but try our preferred peer first, if any. This
* biasses us to the peer that told us of unexpected gossip. */
static struct peer *random_seeker(struct seeker *seeker,
bool (*check_peer)(const struct peer *peer))
{
struct peer *peer = seeker->preferred_peer_softref;
if (peer && check_peer(peer)) {
clear_softref(seeker, &seeker->random_peer_softref);
return peer;
}
return random_peer(seeker->daemon, check_peer);
}
static bool peer_made_progress(struct seeker *seeker) static bool peer_made_progress(struct seeker *seeker)
{ {
const struct peer *peer = seeker->random_peer_softref; const struct peer *peer = seeker->random_peer_softref;
@ -305,7 +334,7 @@ static bool seek_any_unknown_scids(struct seeker *seeker)
return false; return false;
/* No peers can answer? Try again later. */ /* No peers can answer? Try again later. */
peer = random_peer(seeker->daemon, peer_can_take_scid_query); peer = random_seeker(seeker, peer_can_take_scid_query);
if (!peer) if (!peer)
return false; return false;
@ -362,7 +391,7 @@ static bool seek_any_stale_scids(struct seeker *seeker)
return false; return false;
/* No peers can answer? Try again later. */ /* No peers can answer? Try again later. */
peer = random_peer(seeker->daemon, peer_can_take_scid_query); peer = random_seeker(seeker, peer_can_take_scid_query);
if (!peer) if (!peer)
return false; return false;
@ -508,6 +537,9 @@ static void nodeannounce_query_done(struct peer *peer, bool complete)
return; return;
} }
/* Since they told us about new announcements, keep asking them. */
set_preferred_peer(seeker, peer);
/* Double every time. We may skip a few, of course, since map /* Double every time. We may skip a few, of course, since map
* is changing. */ * is changing. */
num_scids *= 2; num_scids *= 2;
@ -532,7 +564,7 @@ static void peer_gossip_probe_nannounces(struct seeker *seeker)
{ {
struct peer *peer; struct peer *peer;
peer = random_peer(seeker->daemon, peer_can_take_scid_query); peer = random_seeker(seeker, peer_can_take_scid_query);
if (!peer) if (!peer)
return; return;
selected_peer(seeker, peer); selected_peer(seeker, peer);
@ -563,7 +595,8 @@ static bool want_update(struct seeker *seeker,
/* They gave us timestamps. Do we want updated versions? */ /* They gave us timestamps. Do we want updated versions? */
static void check_timestamps(struct seeker *seeker, static void check_timestamps(struct seeker *seeker,
struct chan *c, struct chan *c,
const struct channel_update_timestamps *ts) const struct channel_update_timestamps *ts,
struct peer *peer)
{ {
struct stale_scid *stale; struct stale_scid *stale;
u8 query_flag = 0; u8 query_flag = 0;
@ -593,6 +626,7 @@ static void check_timestamps(struct seeker *seeker,
stale->scid = c->scid; stale->scid = c->scid;
stale->query_flag = query_flag; stale->query_flag = query_flag;
stale_scid_map_add(&seeker->stale_scids, stale); stale_scid_map_add(&seeker->stale_scids, stale);
set_preferred_peer(seeker, peer);
} }
} }
@ -615,11 +649,11 @@ static void process_scid_probe(struct peer *peer,
struct chan *c = get_channel(seeker->daemon->rstate, &scids[i]); struct chan *c = get_channel(seeker->daemon->rstate, &scids[i]);
if (c) { if (c) {
if (ts) if (ts)
check_timestamps(seeker, c, ts+i); check_timestamps(seeker, c, ts+i, peer);
continue; continue;
} }
new_unknown_scids |= add_unknown_scid(seeker, &scids[i]); new_unknown_scids |= add_unknown_scid(seeker, &scids[i], peer);
} }
/* No new unknown scids, or no more to ask? We give some wiggle /* No new unknown scids, or no more to ask? We give some wiggle
@ -628,7 +662,7 @@ static void process_scid_probe(struct peer *peer,
&& next_block_range(seeker, number_of_blocks, && next_block_range(seeker, number_of_blocks,
&first_blocknum, &number_of_blocks)) { &first_blocknum, &number_of_blocks)) {
/* This must return a peer, since we have the current peer! */ /* This must return a peer, since we have the current peer! */
peer = random_peer(seeker->daemon, peer_can_take_range_query); peer = random_seeker(seeker, peer_can_take_range_query);
assert(peer); assert(peer);
selected_peer(seeker, peer); selected_peer(seeker, peer);
@ -660,7 +694,7 @@ static void peer_gossip_probe_scids(struct seeker *seeker)
{ {
struct peer *peer; struct peer *peer;
peer = random_peer(seeker->daemon, peer_can_take_range_query); peer = random_seeker(seeker, peer_can_take_range_query);
if (!peer) if (!peer)
return; return;
selected_peer(seeker, peer); selected_peer(seeker, peer);
@ -712,7 +746,7 @@ static void check_firstpeer(struct seeker *seeker)
/* It might have died, pick another. */ /* It might have died, pick another. */
if (!peer) { if (!peer) {
peer = random_peer(seeker->daemon, peer_has_gossip_queries); peer = random_seeker(seeker, peer_has_gossip_queries);
/* No peer? Wait for a new one to join. */ /* No peer? Wait for a new one to join. */
if (!peer) { if (!peer) {
status_debug("seeker: no peers, waiting"); status_debug("seeker: no peers, waiting");
@ -791,7 +825,7 @@ static void maybe_rotate_gossipers(struct seeker *seeker)
size_t i; size_t i;
/* If all peers are gossiping, we're done */ /* If all peers are gossiping, we're done */
peer = random_peer(seeker->daemon, peer_is_not_gossipper); peer = random_seeker(seeker, peer_is_not_gossipper);
if (!peer) if (!peer)
return; return;
@ -895,7 +929,8 @@ bool remove_unknown_scid(struct seeker *seeker,
} }
bool add_unknown_scid(struct seeker *seeker, bool add_unknown_scid(struct seeker *seeker,
const struct short_channel_id *scid) const struct short_channel_id *scid,
struct peer *peer)
{ {
/* Check we're not already getting this one. */ /* Check we're not already getting this one. */
if (scid_map_get(&seeker->unknown_scids, scid)) if (scid_map_get(&seeker->unknown_scids, scid))
@ -903,6 +938,8 @@ bool add_unknown_scid(struct seeker *seeker,
scid_map_add(&seeker->unknown_scids, scid_map_add(&seeker->unknown_scids,
tal_dup(seeker, struct short_channel_id, scid)); tal_dup(seeker, struct short_channel_id, scid));
set_preferred_peer(seeker, peer);
return true; return true;
} }
@ -913,6 +950,6 @@ void query_unknown_channel(struct daemon *daemon,
const struct short_channel_id *id) const struct short_channel_id *id)
{ {
/* Too many, or duplicate? */ /* Too many, or duplicate? */
if (!add_unknown_scid(daemon->seeker, id)) if (!add_unknown_scid(daemon->seeker, id, peer))
return; return;
} }

View File

@ -18,5 +18,6 @@ bool remove_unknown_scid(struct seeker *seeker,
const struct short_channel_id *scid, const struct short_channel_id *scid,
bool found); bool found);
bool add_unknown_scid(struct seeker *seeker, bool add_unknown_scid(struct seeker *seeker,
const struct short_channel_id *scid); const struct short_channel_id *scid,
struct peer *peer);
#endif /* LIGHTNING_GOSSIPD_SEEKER_H */ #endif /* LIGHTNING_GOSSIPD_SEEKER_H */