mirror of
https://github.com/ElementsProject/lightning.git
synced 2025-01-18 05:12:45 +01:00
ef28b6112c
This change is really to allow us to have a --dev-fail-on-subdaemon-fail option so we can handle failures from subdaemons generically. It also neatens handling so we can have an explicit callback for "peer did something wrong" (which matters if we want to close the channel in that case). Signed-off-by: Rusty Russell <rusty@rustcorp.com.au>
284 lines
7.4 KiB
C
284 lines
7.4 KiB
C
#include <ccan/array_size/array_size.h>
|
|
#include <ccan/fdpass/fdpass.h>
|
|
#include <ccan/tal/str/str.h>
|
|
#include <common/cryptomsg.h>
|
|
#include <errno.h>
|
|
#include <fcntl.h>
|
|
#include <handshaked/gen_handshake_wire.h>
|
|
#include <hsmd/gen_hsm_wire.h>
|
|
#include <lightningd/hsm_control.h>
|
|
#include <lightningd/jsonrpc.h>
|
|
#include <lightningd/lightningd.h>
|
|
#include <lightningd/log.h>
|
|
#include <lightningd/new_connection.h>
|
|
#include <lightningd/peer_control.h>
|
|
#include <lightningd/subd.h>
|
|
#include <unistd.h>
|
|
#include <wire/wire_sync.h>
|
|
|
|
const u8 supported_local_features[] = {LOCALFEATURES_INITIAL_ROUTING_SYNC};
|
|
const u8 supported_global_features[] = {0x00};
|
|
|
|
/* Before we have identified the peer, we just have a connection object. */
|
|
struct connection {
|
|
/* Lightning daemon, for when we're handed through callbacks. */
|
|
struct lightningd *ld;
|
|
|
|
/* Where we connected to/from. */
|
|
struct netaddr netaddr;
|
|
|
|
/* Unique identifier for handshaked. */
|
|
u64 unique_id;
|
|
|
|
/* Json command which made us connect (if any) */
|
|
struct command *cmd;
|
|
|
|
/* If we are initiating, we known their id. Otherwise NULL. */
|
|
struct pubkey *known_id;
|
|
};
|
|
|
|
static void connection_destroy(struct connection *c)
|
|
{
|
|
/* FIXME: better diagnostics. */
|
|
if (c->cmd)
|
|
command_fail(c->cmd, "Failed to connect to peer");
|
|
}
|
|
|
|
static void
|
|
PRINTF_FMT(3,4) connection_failed(struct connection *c, struct log *log,
|
|
const char *fmt, ...)
|
|
{
|
|
const char *msg;
|
|
va_list ap;
|
|
|
|
va_start(ap, fmt);
|
|
msg = tal_vfmt(c, fmt, ap);
|
|
va_end(ap);
|
|
log_info(log, "%s", msg);
|
|
if (c->cmd) {
|
|
command_fail(c->cmd, "%s", msg);
|
|
/* Don't fail in destructor, too. */
|
|
c->cmd = NULL;
|
|
}
|
|
tal_free(c);
|
|
}
|
|
|
|
struct connection *new_connection(const tal_t *ctx,
|
|
struct lightningd *ld,
|
|
struct command *cmd,
|
|
const struct pubkey *known_id)
|
|
{
|
|
static u64 id_counter;
|
|
struct connection *c = tal(ctx, struct connection);
|
|
|
|
c->ld = ld;
|
|
c->unique_id = id_counter++;
|
|
c->cmd = cmd;
|
|
if (known_id)
|
|
c->known_id = tal_dup(c, struct pubkey, known_id);
|
|
else
|
|
c->known_id = NULL;
|
|
tal_add_destructor(c, connection_destroy);
|
|
|
|
return c;
|
|
}
|
|
|
|
/**
|
|
* requires_unsupported_features - Check if we support what's being asked
|
|
*
|
|
* Given the features vector that the remote connection is expecting
|
|
* from us, we check to see if we support all even bit features, i.e.,
|
|
* the required features. We do so by subtracting our own features in
|
|
* the provided positions and see if even bits remain.
|
|
*
|
|
* @bitmap: the features bitmap the peer is asking for
|
|
* @supportmap: what do we support
|
|
* @smlen: how long is our supportmap
|
|
*/
|
|
static bool requires_unsupported_features(const u8 *bitmap,
|
|
const u8 *supportmap,
|
|
size_t smlen)
|
|
{
|
|
size_t len = tal_count(bitmap);
|
|
u8 support;
|
|
for (size_t i=0; i<len; i++) {
|
|
/* Find matching bitmap byte in supportmap, 0x00 if none */
|
|
if (len > smlen) {
|
|
support = 0x00;
|
|
} else {
|
|
support = supportmap[smlen-1];
|
|
}
|
|
|
|
/* Cancel out supported bits, check for even bits */
|
|
if ((~support & bitmap[i]) & 0x55)
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
static bool handshake_succeeded(struct subd *handshaked,
|
|
const u8 *msg, const int *fds,
|
|
struct connection *c)
|
|
{
|
|
struct crypto_state cs;
|
|
struct pubkey *id;
|
|
u8 *globalfeatures, *localfeatures;
|
|
|
|
assert(tal_count(fds) == 1);
|
|
|
|
/* FIXME: Look for peer duplicates! */
|
|
|
|
if (!c->known_id) {
|
|
id = tal(msg, struct pubkey);
|
|
if (!fromwire_handshake_responder_reply(c, msg, NULL, id, &cs,
|
|
&globalfeatures,
|
|
&localfeatures))
|
|
goto err;
|
|
log_info(handshaked->log, "Peer in from %s",
|
|
type_to_string(ltmp, struct pubkey, id));
|
|
} else {
|
|
id = c->known_id;
|
|
if (!fromwire_handshake_initiator_reply(c, msg, NULL, &cs,
|
|
&globalfeatures,
|
|
&localfeatures))
|
|
goto err;
|
|
log_info(handshaked->log, "Peer out to %s",
|
|
type_to_string(ltmp, struct pubkey, id));
|
|
}
|
|
|
|
/* BOLT #1:
|
|
*
|
|
* For unknown feature bits which are non-zero, the receiver
|
|
* MUST ignore the bit if the bit number is odd, and MUST fail
|
|
* the connection if the bit number is even.
|
|
*/
|
|
if (requires_unsupported_features(
|
|
globalfeatures, supported_global_features,
|
|
ARRAY_SIZE(supported_global_features))) {
|
|
connection_failed(c, handshaked->log,
|
|
"peer %s: bad globalfeatures: %s",
|
|
type_to_string(c, struct pubkey, id),
|
|
tal_hex(msg, globalfeatures));
|
|
return true;
|
|
}
|
|
|
|
if (requires_unsupported_features(
|
|
localfeatures, supported_local_features,
|
|
ARRAY_SIZE(supported_local_features))) {
|
|
connection_failed(c, handshaked->log,
|
|
"peer %s: bad localfeatures: %s",
|
|
type_to_string(c, struct pubkey, id),
|
|
tal_hex(msg, localfeatures));
|
|
return true;
|
|
}
|
|
|
|
if (c->cmd) {
|
|
struct json_result *response;
|
|
response = new_json_result(c->cmd);
|
|
|
|
json_object_start(response, NULL);
|
|
json_add_pubkey(response, "id", id);
|
|
json_object_end(response);
|
|
command_success(c->cmd, response);
|
|
c->cmd = NULL;
|
|
}
|
|
|
|
add_peer(handshaked->ld, c->unique_id, fds[0], id, &cs);
|
|
/* Now shut handshaked down (frees c as well) */
|
|
return false;
|
|
|
|
err:
|
|
log_broken(handshaked->log, "Malformed resp: %s", tal_hex(c, msg));
|
|
close(fds[0]);
|
|
return false;
|
|
}
|
|
|
|
/* Same path for connecting in vs connecting out. */
|
|
static struct io_plan *hsm_then_handshake(struct io_conn *conn,
|
|
struct lightningd *ld,
|
|
struct connection *c)
|
|
{
|
|
const tal_t *tmpctx = tal_tmpctx(conn);
|
|
int connfd = io_conn_fd(conn), hsmfd;
|
|
struct subd *handshaked;
|
|
u8 *msg;
|
|
|
|
/* Get HSM fd for this peer. */
|
|
msg = towire_hsmctl_hsmfd_ecdh(tmpctx, c->unique_id);
|
|
if (!wire_sync_write(ld->hsm_fd, msg))
|
|
fatal("Could not write to HSM: %s", strerror(errno));
|
|
|
|
msg = hsm_sync_read(tmpctx, ld);
|
|
if (!fromwire_hsmctl_hsmfd_ecdh_fd_reply(msg, NULL))
|
|
fatal("Malformed hsmfd response: %s", tal_hex(msg, msg));
|
|
|
|
hsmfd = fdpass_recv(ld->hsm_fd);
|
|
if (hsmfd < 0)
|
|
fatal("Could not read fd from HSM: %s", strerror(errno));
|
|
|
|
/* Make sure connection fd is blocking */
|
|
io_fd_block(connfd, true);
|
|
|
|
/* Give handshake daemon the hsm fd. */
|
|
handshaked = new_subd(ld, ld,
|
|
"lightning_handshaked", NULL,
|
|
handshake_wire_type_name,
|
|
NULL, NULL, NULL,
|
|
take(&hsmfd), take(&connfd), NULL);
|
|
if (!handshaked) {
|
|
log_unusual(ld->log, "Could not subdaemon handshake: %s",
|
|
strerror(errno));
|
|
goto error;
|
|
}
|
|
|
|
/* If handshake daemon fails, we just drop connection. */
|
|
tal_steal(handshaked, c);
|
|
|
|
if (c->known_id) {
|
|
msg = towire_handshake_initiator(tmpctx, &ld->id, c->known_id);
|
|
} else {
|
|
msg = towire_handshake_responder(tmpctx, &ld->id);
|
|
}
|
|
|
|
/* Now hand peer request to the handshake daemon: hands it
|
|
* back on success */
|
|
subd_req(c, handshaked, take(msg), -1, 1, handshake_succeeded, c);
|
|
|
|
tal_free(tmpctx);
|
|
|
|
/* We don't need conn, we've passed fd to handshaked. */
|
|
return io_close_taken_fd(conn);
|
|
|
|
error:
|
|
close(hsmfd);
|
|
tal_free(tmpctx);
|
|
return io_close(conn);
|
|
}
|
|
|
|
struct io_plan *connection_out(struct io_conn *conn,
|
|
struct lightningd *ld,
|
|
const struct netaddr *netaddr,
|
|
struct connection *c)
|
|
{
|
|
c->netaddr = *netaddr;
|
|
return hsm_then_handshake(conn, ld, c);
|
|
}
|
|
|
|
struct io_plan *connection_in(struct io_conn *conn, struct lightningd *ld)
|
|
{
|
|
struct connection *c = new_connection(ld, ld, NULL, NULL);
|
|
|
|
/* FIXME: Don't assume TCP here. */
|
|
if (!netaddr_from_fd(io_conn_fd(conn), SOCK_STREAM, IPPROTO_TCP,
|
|
&c->netaddr)) {
|
|
log_unusual(ld->log, "Could not get address of incoming fd");
|
|
return io_close(conn);
|
|
}
|
|
return hsm_then_handshake(conn, ld, c);
|
|
}
|
|
|
|
const struct pubkey *connection_known_id(const struct connection *c)
|
|
{
|
|
return c->known_id;
|
|
}
|