/* This contains the code which actively seeks out gossip from peers */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #define GOSSIP_SEEKER_INTERVAL(seeker) \ DEV_FAST_GOSSIP((seeker)->daemon->rstate->dev_fast_gossip, 5, 60) enum seeker_state { /* Still streaming gossip from single peer. */ STARTING_UP, /* Probing: checking our startup really is finished. */ PROBING_SCIDS, /* Probing: check that we have node_announcements. */ PROBING_NANNOUNCES, /* Normal running. */ NORMAL, /* Asking a peer for unknown scids. */ ASKING_FOR_UNKNOWN_SCIDS, /* Asking a peer for stale scids. */ ASKING_FOR_STALE_SCIDS, }; #if DEVELOPER bool dev_suppress_gossip; #endif /* Passthrough helper for HTABLE_DEFINE_TYPE */ static const struct short_channel_id *scid_pass(const struct short_channel_id *s) { return s; } HTABLE_DEFINE_TYPE(struct short_channel_id, scid_pass, hash_scid, short_channel_id_eq, scid_map); /* A channel we have old timestamp(s) for */ struct stale_scid { struct short_channel_id scid; u8 query_flag; }; static const struct short_channel_id *stale_scid_key(const struct stale_scid *s) { return &s->scid; } static bool stale_scid_eq_key(const struct stale_scid *s, const struct short_channel_id *scid) { return short_channel_id_eq(&s->scid, scid); } HTABLE_DEFINE_TYPE(struct stale_scid, stale_scid_key, hash_scid, stale_scid_eq_key, stale_scid_map); /* Gossip we're seeking at the moment. */ struct seeker { struct daemon *daemon; enum seeker_state state; /* Timer which checks on progress every minute */ struct oneshot *check_timer; /* Channels we've heard about, but don't know. */ struct scid_map unknown_scids; /* Channels we've heard about newer timestamps for. */ struct stale_scid_map stale_scids; /* Range of scid blocks we've probed. */ size_t scid_probe_start, scid_probe_end; /* During startup, we ask a single peer for gossip. */ struct peer *random_peer_softref; /* This checks progress of our random peer */ size_t prev_gossip_count; /* Array of scids for node announcements. */ struct short_channel_id *nannounce_scids; u8 *nannounce_query_flags; size_t nannounce_offset; /* Are there any node_ids we didn't know? Implies we're * missing channels. */ bool unknown_nodes; /* Peers we've asked to stream us gossip */ struct peer *gossiper_softref[3]; /* A peer that told us about unknown gossip. */ struct peer *preferred_peer_softref; }; /* Mutual recursion */ static void seeker_check(struct seeker *seeker); static void probe_some_random_scids(struct seeker *seeker); static void begin_check_timer(struct seeker *seeker) { const u32 polltime = GOSSIP_SEEKER_INTERVAL(seeker); seeker->check_timer = new_reltimer(&seeker->daemon->timers, seeker, time_from_sec(polltime), seeker_check, seeker); } #if DEVELOPER static void memleak_help_seeker(struct htable *memtable, struct seeker *seeker) { memleak_remove_htable(memtable, &seeker->unknown_scids.raw); memleak_remove_htable(memtable, &seeker->stale_scids.raw); } #endif /* DEVELOPER */ #define set_state(seeker, state) \ set_state_((seeker), (state), stringify(state)) static void set_state_(struct seeker *seeker, enum seeker_state state, const char *statename) { status_debug("seeker: state = %s", statename); seeker->state = state; } struct seeker *new_seeker(struct daemon *daemon) { struct seeker *seeker = tal(daemon, struct seeker); seeker->daemon = daemon; scid_map_init(&seeker->unknown_scids); stale_scid_map_init(&seeker->stale_scids); seeker->random_peer_softref = NULL; for (size_t i = 0; i < ARRAY_SIZE(seeker->gossiper_softref); i++) seeker->gossiper_softref[i] = NULL; seeker->preferred_peer_softref = NULL; seeker->unknown_nodes = false; set_state(seeker, STARTING_UP); begin_check_timer(seeker); memleak_add_helper(seeker, memleak_help_seeker); return seeker; } /* Set this peer as our random peer; return false if NULL. */ static bool selected_peer(struct seeker *seeker, struct peer *peer) { if (!peer) return false; set_softref(seeker, &seeker->random_peer_softref, peer); status_debug("seeker: chose peer %s", type_to_string(tmpctx, struct node_id, &peer->id)); /* Give it some grace in case we immediately hit timer */ seeker->prev_gossip_count = peer->gossip_counter - GOSSIP_SEEKER_INTERVAL(seeker); 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; /* 80% chance of immediately choosing a peer who reported the missing * stuff: they presumably can tell us more about it. We don't * *always* choose it because it could be simply spamming us with * invalid announcements to get chosen, and we don't handle that case * well yet. */ if (peer && check_peer(peer) && pseudorand(5) != 0) { clear_softref(seeker, &seeker->random_peer_softref); return peer; } return random_peer(seeker->daemon, check_peer); } static bool peer_made_progress(struct seeker *seeker) { const struct peer *peer = seeker->random_peer_softref; /* Has it made progress (at least one valid update per second)? If * not, we assume it's finished, and if it hasn't, we'll end up * querying backwards in next steps. */ if (peer->gossip_counter >= seeker->prev_gossip_count + GOSSIP_SEEKER_INTERVAL(seeker)) { seeker->prev_gossip_count = peer->gossip_counter; return true; } return false; } static void disable_gossip_stream(struct seeker *seeker, struct peer *peer) { u8 *msg; status_debug("seeker: disabling gossip from %s", type_to_string(tmpctx, struct node_id, &peer->id)); /* This is allowed even if they don't understand it (odd) */ msg = towire_gossip_timestamp_filter(NULL, &seeker->daemon->chain_hash, UINT32_MAX, UINT32_MAX); queue_peer_msg(peer, take(msg)); } static void enable_gossip_stream(struct seeker *seeker, struct peer *peer) { /* We seek some way back, to take into account propagation time */ const u32 polltime = GOSSIP_SEEKER_INTERVAL(seeker) * 10; u32 start = seeker->daemon->rstate->last_timestamp; u8 *msg; #if DEVELOPER if (dev_suppress_gossip) return; #endif if (start > polltime) start -= polltime; else start = 0; status_debug("seeker: starting gossip from %s", type_to_string(tmpctx, struct node_id, &peer->id)); /* This is allowed even if they don't understand it (odd) */ msg = towire_gossip_timestamp_filter(NULL, &seeker->daemon->chain_hash, start, UINT32_MAX); queue_peer_msg(peer, take(msg)); } static void normal_gossip_start(struct seeker *seeker, struct peer *peer) { bool enable_stream = false; /* Make this one of our streaming gossipers if we aren't full */ for (size_t i = 0; i < ARRAY_SIZE(seeker->gossiper_softref); i++) { if (seeker->gossiper_softref[i] == NULL) { set_softref(seeker, &seeker->gossiper_softref[i], peer); enable_stream = true; break; } } if (enable_stream) enable_gossip_stream(seeker, peer); else disable_gossip_stream(seeker, peer); } /* Turn unknown_scids map into a flat array, removes from map. */ static struct short_channel_id *unknown_scids_remove(const tal_t *ctx, struct seeker *seeker) { struct scid_map *map = &seeker->unknown_scids; struct short_channel_id *scids, *s; size_t i, max; struct scid_map_iter it; /* Marshal into an array: we can fit 8000 comfortably. */ if (scid_map_count(map) < 8000) max = scid_map_count(map); else max = 8000; scids = tal_arr(ctx, struct short_channel_id, max); for (i = 0, s = scid_map_first(map, &it); i < max; i++) { scids[i] = *s; scid_map_del(map, s); tal_free(s); } assert(i == tal_count(scids)); return scids; } /* We have selected this peer to stream us startup gossip */ static void peer_gossip_startup(struct seeker *seeker, struct peer *peer) { selected_peer(seeker, peer); normal_gossip_start(seeker, peer); } static bool peer_has_gossip_queries(const struct peer *peer) { return peer->gossip_queries_feature; } static bool peer_can_take_range_query(const struct peer *peer) { return peer->gossip_queries_feature && !peer->query_channel_blocks; } static bool peer_can_take_scid_query(const struct peer *peer) { return peer->gossip_queries_feature && !peer->scid_query_outstanding; } static void scid_query_done(struct peer *peer, bool complete) { struct seeker *seeker = peer->daemon->seeker; /* Peer completed! OK, start random scid probe in case we're * still missing gossip. */ probe_some_random_scids(seeker); } /* Returns true if there were scids to seek. */ static bool seek_any_unknown_scids(struct seeker *seeker) { struct peer *peer; struct short_channel_id *scids; /* Nothing we need to know about? */ if (scid_map_count(&seeker->unknown_scids) == 0) return false; /* No peers can answer? Try again later. */ peer = random_seeker(seeker, peer_can_take_scid_query); if (!peer) return false; set_state(seeker, ASKING_FOR_UNKNOWN_SCIDS); selected_peer(seeker, peer); scids = unknown_scids_remove(tmpctx, seeker); if (!query_short_channel_ids(seeker->daemon, peer, scids, NULL, scid_query_done)) status_failed(STATUS_FAIL_INTERNAL_ERROR, "seeker: quering %zu scids is too many?", tal_count(scids)); return true; } /* Turns stale_scid_map into two arrays, and removes from map */ static struct short_channel_id *stale_scids_remove(const tal_t *ctx, struct seeker *seeker, u8 **query_flags) { struct stale_scid_map *map = &seeker->stale_scids; struct short_channel_id *scids; const struct stale_scid *s; size_t i, max; struct stale_scid_map_iter it; /* Marshal into an array: we can fit 7000 comfortably (8 byte scid, 1 byte flag). */ if (stale_scid_map_count(map) < 7000) max = stale_scid_map_count(map); else max = 7000; scids = tal_arr(ctx, struct short_channel_id, max); *query_flags = tal_arr(ctx, u8, max); for (i = 0, s = stale_scid_map_first(map, &it); i < max; i++) { scids[i] = s->scid; (*query_flags)[i] = s->query_flag; stale_scid_map_del(map, s); tal_free(s); } assert(i == tal_count(scids)); return scids; } static bool seek_any_stale_scids(struct seeker *seeker) { struct peer *peer; struct short_channel_id *scids; u8 *query_flags; /* Nothing we need to know about? */ if (stale_scid_map_count(&seeker->stale_scids) == 0) return false; /* No peers can answer? Try again later. */ peer = random_seeker(seeker, peer_can_take_scid_query); if (!peer) return false; set_state(seeker, ASKING_FOR_STALE_SCIDS); selected_peer(seeker, peer); /* This is best-effort, so this consumes them as well. */ scids = stale_scids_remove(tmpctx, seeker, &query_flags); if (!query_short_channel_ids(seeker->daemon, peer, scids, query_flags, scid_query_done)) status_failed(STATUS_FAIL_INTERNAL_ERROR, "seeker: quering %zu scids is too many?", tal_count(scids)); return true; } /* We can't ask for channels by node_id, so probe at random */ /* Returns true and sets first_blocknum and number_of_blocks if * there's more to find. */ static bool next_block_range(struct seeker *seeker, u32 prev_num_blocks, u32 *first_blocknum, u32 *number_of_blocks) { const u32 current_height = seeker->daemon->current_blockheight; /* We always try to get twice as many as last time. */ *number_of_blocks = prev_num_blocks * 2; if (seeker->scid_probe_start > 0) { /* Enlarge probe to cover prior blocks, but twice as many. */ if (*number_of_blocks > seeker->scid_probe_start) { *number_of_blocks = seeker->scid_probe_start; *first_blocknum = 0; } else { *first_blocknum = seeker->scid_probe_start - *number_of_blocks; } seeker->scid_probe_start = *first_blocknum; return true; } /* We allow 6 new blocks since we started; they should be empty anyway */ if (seeker->scid_probe_end + 6 < current_height) { if (seeker->scid_probe_end + *number_of_blocks > current_height) *number_of_blocks = current_height - seeker->scid_probe_end; *first_blocknum = seeker->scid_probe_end + 1; seeker->scid_probe_end = *first_blocknum + *number_of_blocks - 1; return true; } /* No more to find. */ return false; } static bool get_unannounced_nodes(const tal_t *ctx, struct routing_state *rstate, size_t off, size_t max, struct short_channel_id **scids, u8 **query_flags) { struct node_map_iter it; size_t i = 0, num = 0; struct node *n; /* Pick an example short_channel_id at random to query. As a * side-effect this gets the node */ *scids = tal_arr(ctx, struct short_channel_id, max); *query_flags = tal_arr(ctx, u8, max); for (n = node_map_first(rstate->nodes, &it); n && num < max; n = node_map_next(rstate->nodes, &it)) { if (n->bcast.index) continue; if (i >= off) { struct chan_map_iter cit; struct chan *c = first_chan(n, &cit); (*scids)[num] = c->scid; if (c->nodes[0] == n) (*query_flags)[num] = SCID_QF_NODE1; else (*query_flags)[num] = SCID_QF_NODE2; num++; } i++; } if (num == 0) { *scids = tal_free(*scids); *query_flags = tal_free(*query_flags); return false; } if (num < max) { tal_resize(scids, num); tal_resize(query_flags, i - off); } return true; } /* Mutual recursion */ static void peer_gossip_probe_nannounces(struct seeker *seeker); static void nodeannounce_query_done(struct peer *peer, bool complete) { struct seeker *seeker = peer->daemon->seeker; struct routing_state *rstate = seeker->daemon->rstate; size_t new_nannounce = 0, num_scids; /* We might have given up on them, then they replied. */ if (seeker->random_peer_softref != peer) return; clear_softref(seeker, &seeker->random_peer_softref); num_scids = tal_count(seeker->nannounce_scids); for (size_t i = 0; i < num_scids; i++) { struct chan *c = get_channel(rstate, &seeker->nannounce_scids[i]); /* Could have closed since we asked. */ if (!c) continue; if ((seeker->nannounce_query_flags[i] & SCID_QF_NODE1) && c->nodes[0]->bcast.index) new_nannounce++; if ((seeker->nannounce_query_flags[i] & SCID_QF_NODE2) && c->nodes[1]->bcast.index) new_nannounce++; } status_debug("seeker: found %zu new node_announcements in %zu scids", new_nannounce, num_scids); seeker->nannounce_scids = tal_free(seeker->nannounce_scids); seeker->nannounce_query_flags = tal_free(seeker->nannounce_query_flags); seeker->nannounce_offset += num_scids; if (!new_nannounce) { set_state(seeker, NORMAL); 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 * is changing. */ num_scids *= 2; /* Don't try to create a query larger than 64k */ if (num_scids > 7000) num_scids = 7000; if (!get_unannounced_nodes(seeker, seeker->daemon->rstate, seeker->nannounce_offset, num_scids, &seeker->nannounce_scids, &seeker->nannounce_query_flags)) { /* Nothing unknown at all? Great, we're done */ set_state(seeker, NORMAL); return; } peer_gossip_probe_nannounces(seeker); } /* Pick a peer, ask it for a few node announcements, to check. */ static void peer_gossip_probe_nannounces(struct seeker *seeker) { struct peer *peer; set_state(seeker, PROBING_NANNOUNCES); peer = random_seeker(seeker, peer_can_take_scid_query); if (!peer) return; selected_peer(seeker, peer); if (!query_short_channel_ids(seeker->daemon, peer, seeker->nannounce_scids, seeker->nannounce_query_flags, nodeannounce_query_done)) status_failed(STATUS_FAIL_INTERNAL_ERROR, "seeker: quering %zu scids is too many?", tal_count(seeker->nannounce_scids)); } /* They have update with this timestamp: do we want it? */ static bool want_update(struct seeker *seeker, u32 timestamp, const struct half_chan *hc) { if (!is_halfchan_defined(hc)) return timestamp != 0; if (timestamp <= hc->bcast.timestamp) return false; return !would_ratelimit_cupdate(seeker->daemon->rstate, hc, timestamp); } /* They gave us timestamps. Do we want updated versions? */ static void check_timestamps(struct seeker *seeker, struct chan *c, const struct channel_update_timestamps *ts, struct peer *peer) { struct stale_scid *stale; u8 query_flag = 0; /* BOLT #7: * * `timestamp_node_id_1` is the timestamp of the `channel_update` * for `node_id_1`, or 0 if there was no `channel_update` from that * node. * * `timestamp_node_id_2` is the timestamp of the `channel_update` * for `node_id_2`, or 0 if there was no `channel_update` from that * node. */ if (want_update(seeker, ts->timestamp_node_id_1, &c->half[0])) query_flag |= SCID_QF_UPDATE1; if (want_update(seeker, ts->timestamp_node_id_2, &c->half[1])) query_flag |= SCID_QF_UPDATE2; if (!query_flag) return; /* Add in flags if we're already getting it. */ stale = stale_scid_map_get(&seeker->stale_scids, &c->scid); if (stale) stale->query_flag |= query_flag; else { stale = tal(seeker, struct stale_scid); stale->scid = c->scid; stale->query_flag = query_flag; stale_scid_map_add(&seeker->stale_scids, stale); set_preferred_peer(seeker, peer); } } static void process_scid_probe(struct peer *peer, u32 first_blocknum, u32 number_of_blocks, const struct short_channel_id *scids, const struct channel_update_timestamps *ts, bool complete) { struct seeker *seeker = peer->daemon->seeker; bool new_unknown_scids = false; /* We might have given up on them, then they replied. */ if (seeker->random_peer_softref != peer) return; clear_softref(seeker, &seeker->random_peer_softref); for (size_t i = 0; i < tal_count(scids); i++) { struct chan *c = get_channel(seeker->daemon->rstate, &scids[i]); if (c) { if (ts) check_timestamps(seeker, c, ts+i, peer); continue; } new_unknown_scids |= add_unknown_scid(seeker, &scids[i], peer); } /* No new unknown scids, or no more to ask? We give some wiggle * room in case blocks came in since we started. */ if (new_unknown_scids && next_block_range(seeker, number_of_blocks, &first_blocknum, &number_of_blocks)) { /* This must return a peer, since we have the current peer! */ peer = random_seeker(seeker, peer_can_take_range_query); assert(peer); selected_peer(seeker, peer); query_channel_range(seeker->daemon, peer, first_blocknum, number_of_blocks, QUERY_ADD_TIMESTAMPS, process_scid_probe); return; } /* Channel probe finished, try asking for 32 unannounced nodes. */ seeker->nannounce_offset = 0; if (!get_unannounced_nodes(seeker, seeker->daemon->rstate, seeker->nannounce_offset, 32, &seeker->nannounce_scids, &seeker->nannounce_query_flags)) { /* No unknown nodes. Great! */ set_state(seeker, NORMAL); return; } peer_gossip_probe_nannounces(seeker); } /* Pick a peer, ask it for a few scids, to check. */ static void peer_gossip_probe_scids(struct seeker *seeker) { struct peer *peer; set_state(seeker, PROBING_SCIDS); peer = random_seeker(seeker, peer_can_take_range_query); if (!peer) return; selected_peer(seeker, peer); /* This calls process_scid_probe when we get the reply. */ query_channel_range(seeker->daemon, peer, seeker->scid_probe_start, seeker->scid_probe_end - seeker->scid_probe_start + 1, QUERY_ADD_TIMESTAMPS, process_scid_probe); } static void probe_random_scids(struct seeker *seeker, size_t num_blocks) { if (seeker->daemon->current_blockheight < num_blocks) { seeker->scid_probe_start = 0; seeker->scid_probe_end = seeker->daemon->current_blockheight; } else { seeker->scid_probe_start = pseudorand(seeker->daemon->current_blockheight + num_blocks); seeker->scid_probe_end = seeker->scid_probe_start + num_blocks - 1; } seeker->nannounce_scids = NULL; seeker->nannounce_offset = 0; peer_gossip_probe_scids(seeker); } /* We usually get a channel per block, so these cover a fair bit of ground */ static void probe_some_random_scids(struct seeker *seeker) { return probe_random_scids(seeker, 64); } static void probe_many_random_scids(struct seeker *seeker) { return probe_random_scids(seeker, 1008); } static void check_firstpeer(struct seeker *seeker) { struct chan *c; u64 index; struct peer *peer = seeker->random_peer_softref, *p; /* It might have died, pick another. */ if (!peer) { peer = random_seeker(seeker, peer_has_gossip_queries); /* No peer? Wait for a new one to join. */ if (!peer) { status_debug("seeker: no peers, waiting"); return; } peer_gossip_startup(seeker, peer); return; } /* If no progress, we assume it's finished, and if it hasn't, * we'll end up querying backwards in next steps. */ if (peer_made_progress(seeker)) return; /* Other peers can gossip now. */ status_debug("seeker: startup peer finished"); clear_softref(seeker, &seeker->random_peer_softref); list_for_each(&seeker->daemon->peers, p, list) { if (p == peer) continue; normal_gossip_start(seeker, p); } /* We always look up 6 prior to last we have */ c = uintmap_last(&seeker->daemon->rstate->chanmap, &index); if (c && short_channel_id_blocknum(&c->scid) > 6) { seeker->scid_probe_start = short_channel_id_blocknum(&c->scid) - 6; } else { seeker->scid_probe_start = 0; } seeker->scid_probe_end = seeker->daemon->current_blockheight; peer_gossip_probe_scids(seeker); } static void check_probe(struct seeker *seeker, void (*restart)(struct seeker *seeker)) { struct peer *peer = seeker->random_peer_softref; /* It might have died, pick another. */ if (!peer) { restart(seeker); return; } /* Is peer making progress with responses? */ if (peer_made_progress(seeker)) return; status_debug("Peer %s has only moved gossip %zu->%zu for probe, giving up on it", type_to_string(tmpctx, struct node_id, &peer->id), seeker->prev_gossip_count, peer->gossip_counter); clear_softref(seeker, &seeker->random_peer_softref); restart(seeker); } static bool peer_is_not_gossipper(const struct peer *peer) { const struct seeker *seeker = peer->daemon->seeker; for (size_t i = 0; i < ARRAY_SIZE(seeker->gossiper_softref); i++) { if (seeker->gossiper_softref[i] == peer) return false; } return true; } /* FIXME: We should look at gossip performance and replace the underperforming * peers in preference. */ static void maybe_rotate_gossipers(struct seeker *seeker) { struct peer *peer; size_t i; /* If all peers are gossiping, we're done */ peer = random_seeker(seeker, peer_is_not_gossipper); if (!peer) return; /* If we have a slot free, or ~ 1 per hour */ for (i = 0; i < ARRAY_SIZE(seeker->gossiper_softref); i++) { if (!seeker->gossiper_softref[i]) goto set_gossiper; if (pseudorand(ARRAY_SIZE(seeker->gossiper_softref) * 60) == 0) goto clear_and_set_gossiper; } return; clear_and_set_gossiper: disable_gossip_stream(seeker, seeker->gossiper_softref[i]); clear_softref(seeker, &seeker->gossiper_softref[i]); set_gossiper: set_softref(seeker, &seeker->gossiper_softref[i], peer); enable_gossip_stream(seeker, peer); } static bool seek_any_unknown_nodes(struct seeker *seeker) { if (!seeker->unknown_nodes) return false; seeker->unknown_nodes = false; probe_many_random_scids(seeker); return true; } /* Periodic timer to see how our gossip is going. */ static void seeker_check(struct seeker *seeker) { #if DEVELOPER if (dev_suppress_gossip) return; #endif switch (seeker->state) { case STARTING_UP: check_firstpeer(seeker); break; case PROBING_SCIDS: check_probe(seeker, peer_gossip_probe_scids); break; case ASKING_FOR_UNKNOWN_SCIDS: check_probe(seeker, probe_many_random_scids); break; case ASKING_FOR_STALE_SCIDS: check_probe(seeker, probe_some_random_scids); break; case PROBING_NANNOUNCES: check_probe(seeker, peer_gossip_probe_nannounces); break; case NORMAL: maybe_rotate_gossipers(seeker); if (!seek_any_unknown_scids(seeker) && !seek_any_stale_scids(seeker)) seek_any_unknown_nodes(seeker); break; } begin_check_timer(seeker); } /* We get this when we have a new peer. */ void seeker_setup_peer_gossip(struct seeker *seeker, struct peer *peer) { bool have_probing_peer = seeker->random_peer_softref != NULL; /* Can't do anything useful with these peers. */ if (!peer->gossip_queries_feature) return; #if DEVELOPER if (dev_suppress_gossip) return; #endif switch (seeker->state) { case STARTING_UP: if (!have_probing_peer) peer_gossip_startup(seeker, peer); /* Waiting for seeker_check to release us */ return; /* In these states, we set up peers to stream gossip normally */ case PROBING_SCIDS: if (!have_probing_peer) peer_gossip_probe_scids(seeker); goto start_them_gossiping; case PROBING_NANNOUNCES: if (!have_probing_peer) peer_gossip_probe_nannounces(seeker); goto start_them_gossiping; case NORMAL: case ASKING_FOR_UNKNOWN_SCIDS: case ASKING_FOR_STALE_SCIDS: goto start_them_gossiping; } abort(); start_them_gossiping: normal_gossip_start(seeker, peer); } bool remove_unknown_scid(struct seeker *seeker, const struct short_channel_id *scid, bool found /*FIXME: use this info!*/) { struct short_channel_id *unknown; unknown = scid_map_get(&seeker->unknown_scids, scid); if (unknown) { scid_map_del(&seeker->unknown_scids, unknown); tal_free(unknown); return true; } return false; } bool add_unknown_scid(struct seeker *seeker, const struct short_channel_id *scid, struct peer *peer) { /* Check we're not already getting this one. */ if (scid_map_get(&seeker->unknown_scids, scid)) return false; scid_map_add(&seeker->unknown_scids, tal_dup(seeker, struct short_channel_id, scid)); set_preferred_peer(seeker, peer); return true; } /* This peer told us about an update to an unknown channel. Ask it for a * channel_announcement. */ void query_unknown_channel(struct daemon *daemon, struct peer *peer, const struct short_channel_id *id) { /* Too many, or duplicate? */ if (!add_unknown_scid(daemon->seeker, id, peer)) return; } /* This peer told us about an unknown node. Start probing it. */ void query_unknown_node(struct seeker *seeker, struct peer *peer) { seeker->unknown_nodes = true; set_preferred_peer(seeker, peer); }