Browse Source

gossipd: take connectd fd on initialization.

connectd has a dedicated fd to gossipd, so it can ask for a new gossip_fd
for a peer.

gossipd has a standalone routine to create a remote peer (this will
eventually be the only way gossipd creates a new peer).

For now lightningd creates a socketpair but doesn't run connectd, so
gossipd never sees any requests on this fd.

Signed-off-by: Rusty Russell <rusty@rustcorp.com.au>
ppa-0.6.1
Rusty Russell 7 years ago
parent
commit
92d66a5451
  1. 1
      gossipd/Makefile
  2. 129
      gossipd/gossip.c
  3. 11
      lightningd/connect_control.c
  4. 2
      lightningd/connect_control.h
  5. 4
      lightningd/gossip_control.c
  6. 2
      lightningd/gossip_control.h
  7. 7
      lightningd/lightningd.c
  8. 5
      lightningd/test/run-find_my_path.c

1
gossipd/Makefile

@ -66,6 +66,7 @@ GOSSIPD_COMMON_OBJS := \
common/version.o \ common/version.o \
common/wireaddr.o \ common/wireaddr.o \
common/wire_error.o \ common/wire_error.o \
connectd/gen_connect_gossip_wire.o \
hsmd/client.o \ hsmd/client.o \
hsmd/gen_hsm_client_wire.o \ hsmd/gen_hsm_client_wire.o \
lightningd/gossip_msg.o \ lightningd/gossip_msg.o \

129
gossipd/gossip.c

@ -30,6 +30,7 @@
#include <common/version.h> #include <common/version.h>
#include <common/wire_error.h> #include <common/wire_error.h>
#include <common/wireaddr.h> #include <common/wireaddr.h>
#include <connectd/gen_connect_gossip_wire.h>
#include <errno.h> #include <errno.h>
#include <fcntl.h> #include <fcntl.h>
#include <gossipd/broadcast.h> #include <gossipd/broadcast.h>
@ -61,6 +62,7 @@
#define GOSSIP_MAX_REACH_ATTEMPTS 10 #define GOSSIP_MAX_REACH_ATTEMPTS 10
#define HSM_FD 3 #define HSM_FD 3
#define CONNECTD_FD 4
#define INITIAL_WAIT_SECONDS 1 #define INITIAL_WAIT_SECONDS 1
#define MAX_WAIT_SECONDS 300 #define MAX_WAIT_SECONDS 300
@ -141,6 +143,9 @@ struct daemon {
/* Connection to main daemon. */ /* Connection to main daemon. */
struct daemon_conn master; struct daemon_conn master;
/* Connection to connect daemon. */
struct daemon_conn connectd;
/* Routing information */ /* Routing information */
struct routing_state *rstate; struct routing_state *rstate;
@ -710,6 +715,7 @@ static struct io_plan *peer_connected(struct io_conn *conn, struct peer *peer)
wake_gossip_out(peer); wake_gossip_out(peer);
setup_gossip_range(peer); setup_gossip_range(peer);
return io_close_taken_fd(conn); return io_close_taken_fd(conn);
} }
@ -1948,6 +1954,107 @@ static bool send_peer_with_fds(struct peer *peer, const u8 *msg)
return true; return true;
} }
static void free_peer(struct io_conn *conn, struct daemon_conn *dc)
{
struct peer *peer = dc->ctx;
tal_free(peer);
}
static struct io_plan *connectd_new_peer(struct io_conn *conn,
struct daemon *daemon,
const u8 *msg)
{
struct peer *peer = tal(conn, struct peer);
int fds[2];
if (!fromwire_gossip_new_peer(msg, &peer->id,
&peer->gossip_queries_feature,
&peer->initial_routing_sync_feature)) {
status_broken("Bad new_peer msg from connectd: %s",
tal_hex(tmpctx, msg));
return io_close(conn);
}
if (socketpair(AF_LOCAL, SOCK_STREAM, 0, fds) != 0) {
status_broken("Failed to create socketpair: %s",
strerror(errno));
daemon_conn_send(&daemon->connectd,
take(towire_gossip_new_peer_reply(NULL, false)));
goto done;
}
/* We might not have noticed old peer is dead; kill it now. */
tal_free(find_peer(daemon, &peer->id));
/* FIXME: Remove addr field. */
peer->addr.itype = ADDR_INTERNAL_WIREADDR;
peer->addr.u.wireaddr.type = ADDR_TYPE_PADDING;
/* FIXME: Remove these fields. */
peer->lfeatures = peer->gfeatures = NULL;
peer->daemon = daemon;
peer->remote = tal(peer, struct daemon_conn);
daemon_conn_init(peer, peer->remote, fds[0], owner_msg_in, free_peer);
peer->remote->msg_queue_cleared_cb = nonlocal_dump_gossip;
peer->scid_queries = NULL;
peer->scid_query_idx = 0;
peer->scid_query_nodes = NULL;
peer->scid_query_nodes_idx = 0;
peer->num_scid_queries_outstanding = 0;
peer->query_channel_blocks = NULL;
peer->num_pings_outstanding = 0;
peer->gossip_timer = NULL;
list_add_tail(&peer->daemon->peers, &peer->list);
tal_add_destructor(peer, destroy_peer);
/* BOLT #7:
*
* - if the `gossip_queries` feature is negotiated:
* - MUST NOT relay any gossip messages unless explicitly requested.
*/
if (peer->gossip_queries_feature) {
peer->broadcast_index = UINT64_MAX;
/* Nothing in this range */
peer->gossip_timestamp_min = UINT32_MAX;
peer->gossip_timestamp_max = 0;
} else {
/* BOLT #7:
*
* - upon receiving an `init` message with the
* `initial_routing_sync` flag set to 1:
* - SHOULD send gossip messages for all known channels and
* nodes, as if they were just received.
* - if the `initial_routing_sync` flag is set to 0, OR if the
* initial sync was completed:
* - SHOULD resume normal operation, as specified in the
* following [Rebroadcasting](#rebroadcasting) section.
*/
peer->gossip_timestamp_min = 0;
peer->gossip_timestamp_max = UINT32_MAX;
if (peer->initial_routing_sync_feature)
peer->broadcast_index = 0;
else
peer->broadcast_index
= peer->daemon->rstate->broadcasts->next_index;
}
/* Start the gossip flowing. */
wake_gossip_out(peer);
setup_gossip_range(peer);
/* Reply with success, and the new fd */
daemon_conn_send(&daemon->connectd,
take(towire_gossip_new_peer_reply(NULL, true)));
daemon_conn_send_fd(&daemon->connectd, fds[1]);
done:
return daemon_conn_read_next(conn, &daemon->connectd);
}
/** /**
* nonlocal_dump_gossip - catch the nonlocal peer up with the latest gossip. * nonlocal_dump_gossip - catch the nonlocal peer up with the latest gossip.
* *
@ -3809,6 +3916,26 @@ static struct io_plan *recv_req(struct io_conn *conn, struct daemon_conn *master
t, tal_hex(tmpctx, master->msg_in)); t, tal_hex(tmpctx, master->msg_in));
} }
static struct io_plan *connectd_req(struct io_conn *conn,
struct daemon_conn *connectd)
{
struct daemon *daemon = container_of(connectd, struct daemon, connectd);
enum connect_gossip_wire_type t = fromwire_peektype(connectd->msg_in);
switch (t) {
case WIRE_GOSSIP_NEW_PEER:
return connectd_new_peer(conn, daemon, connectd->msg_in);
/* We send this, don't receive it. */
case WIRE_GOSSIP_NEW_PEER_REPLY:
break;
}
status_broken("Bad msg from connectd: %s",
tal_hex(tmpctx, connectd->msg_in));
return io_close(conn);
}
#ifndef TESTING #ifndef TESTING
static void master_gone(struct io_conn *unused UNUSED, struct daemon_conn *dc UNUSED) static void master_gone(struct io_conn *unused UNUSED, struct daemon_conn *dc UNUSED)
{ {
@ -3840,6 +3967,8 @@ int main(int argc, char *argv[])
master_gone); master_gone);
status_setup_async(&daemon->master); status_setup_async(&daemon->master);
hsm_setup(HSM_FD); hsm_setup(HSM_FD);
daemon_conn_init(daemon, &daemon->connectd, CONNECTD_FD, connectd_req,
NULL);
/* When conn closes, everything is freed. */ /* When conn closes, everything is freed. */
tal_steal(daemon->master.conn, daemon); tal_steal(daemon->master.conn, daemon);

11
lightningd/connect_control.c

@ -188,3 +188,14 @@ static const struct json_command connect_command = {
"{id} can also be of the form id@host" "{id} can also be of the form id@host"
}; };
AUTODATA(json_command, &connect_command); AUTODATA(json_command, &connect_command);
int connectd_init(struct lightningd *ld)
{
/* FIXME: implement */
int fds[2];
if (socketpair(AF_LOCAL, SOCK_STREAM, 0, fds) != 0)
fatal("Could not socketpair for connectd<->gossipd");
return fds[0];
}

2
lightningd/connect_control.h

@ -5,6 +5,8 @@
struct lightningd; struct lightningd;
struct pubkey; struct pubkey;
/* Returns fd for gossipd to talk to connectd */
int connectd_init(struct lightningd *ld);
void gossip_connect_result(struct lightningd *ld, const u8 *msg); void gossip_connect_result(struct lightningd *ld, const u8 *msg);
#endif /* LIGHTNING_LIGHTNINGD_CONNECT_CONTROL_H */ #endif /* LIGHTNING_LIGHTNINGD_CONNECT_CONTROL_H */

4
lightningd/gossip_control.c

@ -199,7 +199,7 @@ static unsigned gossip_msg(struct subd *gossip, const u8 *msg, const int *fds)
/* Create the `gossipd` subdaemon and send the initialization /* Create the `gossipd` subdaemon and send the initialization
* message */ * message */
void gossip_init(struct lightningd *ld) void gossip_init(struct lightningd *ld, int connectd_fd)
{ {
u8 *msg; u8 *msg;
int hsmfd; int hsmfd;
@ -226,7 +226,7 @@ void gossip_init(struct lightningd *ld)
ld->gossip = new_global_subd(ld, "lightning_gossipd", ld->gossip = new_global_subd(ld, "lightning_gossipd",
gossip_wire_type_name, gossip_msg, gossip_wire_type_name, gossip_msg,
take(&hsmfd), NULL); take(&hsmfd), take(&connectd_fd), NULL);
if (!ld->gossip) if (!ld->gossip)
err(1, "Could not subdaemon gossip"); err(1, "Could not subdaemon gossip");

2
lightningd/gossip_control.h

@ -7,7 +7,7 @@
struct lightningd; struct lightningd;
void gossip_init(struct lightningd *ld); void gossip_init(struct lightningd *ld, int connectd_fd);
void gossip_activate(struct lightningd *ld); void gossip_activate(struct lightningd *ld);
void gossipd_notify_spend(struct lightningd *ld, void gossipd_notify_spend(struct lightningd *ld,

7
lightningd/lightningd.c

@ -28,6 +28,7 @@
#include <lightningd/bitcoind.h> #include <lightningd/bitcoind.h>
#include <lightningd/chaintopology.h> #include <lightningd/chaintopology.h>
#include <lightningd/channel_control.h> #include <lightningd/channel_control.h>
#include <lightningd/connect_control.h>
#include <lightningd/invoice.h> #include <lightningd/invoice.h>
#include <lightningd/jsonrpc.h> #include <lightningd/jsonrpc.h>
#include <lightningd/log.h> #include <lightningd/log.h>
@ -310,6 +311,7 @@ int main(int argc, char *argv[])
{ {
struct lightningd *ld; struct lightningd *ld;
u32 min_blockheight, max_blockheight; u32 min_blockheight, max_blockheight;
int connectd_gossipd_fd;
setup_locale(); setup_locale();
daemon_setup(argv[0], log_backtrace_print, log_backtrace_exit); daemon_setup(argv[0], log_backtrace_print, log_backtrace_exit);
@ -345,8 +347,11 @@ int main(int argc, char *argv[])
/* Now we know our ID, we can set our color/alias if not already. */ /* Now we know our ID, we can set our color/alias if not already. */
setup_color_and_alias(ld); setup_color_and_alias(ld);
/* Set up connect daemon. */
connectd_gossipd_fd = connectd_init(ld);
/* Set up gossip daemon. */ /* Set up gossip daemon. */
gossip_init(ld); gossip_init(ld, connectd_gossipd_fd);
/* Everything is within a transaction. */ /* Everything is within a transaction. */
db_begin_transaction(ld->wallet->db); db_begin_transaction(ld->wallet->db);

5
lightningd/test/run-find_my_path.c

@ -15,6 +15,9 @@ void begin_topology(struct chain_topology *topo UNNEEDED)
void channel_notify_new_block(struct lightningd *ld UNNEEDED, void channel_notify_new_block(struct lightningd *ld UNNEEDED,
u32 block_height UNNEEDED) u32 block_height UNNEEDED)
{ fprintf(stderr, "channel_notify_new_block called!\n"); abort(); } { fprintf(stderr, "channel_notify_new_block called!\n"); abort(); }
/* Generated stub for connectd_init */
int connectd_init(struct lightningd *ld UNNEEDED)
{ fprintf(stderr, "connectd_init called!\n"); abort(); }
/* Generated stub for daemon_setup */ /* Generated stub for daemon_setup */
void daemon_setup(const char *argv0 UNNEEDED, void daemon_setup(const char *argv0 UNNEEDED,
void (*backtrace_print)(const char *fmt UNNEEDED, ...) UNNEEDED, void (*backtrace_print)(const char *fmt UNNEEDED, ...) UNNEEDED,
@ -51,7 +54,7 @@ void free_htlcs(struct lightningd *ld UNNEEDED, const struct channel *channel UN
void gossip_activate(struct lightningd *ld UNNEEDED) void gossip_activate(struct lightningd *ld UNNEEDED)
{ fprintf(stderr, "gossip_activate called!\n"); abort(); } { fprintf(stderr, "gossip_activate called!\n"); abort(); }
/* Generated stub for gossip_init */ /* Generated stub for gossip_init */
void gossip_init(struct lightningd *ld UNNEEDED) void gossip_init(struct lightningd *ld UNNEEDED, int connectd_fd UNNEEDED)
{ fprintf(stderr, "gossip_init called!\n"); abort(); } { fprintf(stderr, "gossip_init called!\n"); abort(); }
/* Generated stub for handle_opts */ /* Generated stub for handle_opts */
void handle_opts(struct lightningd *ld UNNEEDED, int argc UNNEEDED, char *argv[]) void handle_opts(struct lightningd *ld UNNEEDED, int argc UNNEEDED, char *argv[])

Loading…
Cancel
Save