Browse Source

daemon: socket code.

At the moment, if you connect it just says Hello! and closes the socket.

Signed-off-by: Rusty Russell <rusty@rustcorp.com.au>
ppa-0.6.1
Rusty Russell 9 years ago
parent
commit
469401610f
  1. 2
      daemon/Makefile
  2. 9
      daemon/lightningd.c
  3. 4
      daemon/lightningd.h
  4. 178
      daemon/peer.c
  5. 43
      daemon/peer.h

2
daemon/Makefile

@ -16,6 +16,7 @@ DAEMON_LIB_OBJS := $(DAEMON_LIB_SRC:.c=.o)
DAEMON_SRC := \ DAEMON_SRC := \
daemon/jsonrpc.c \ daemon/jsonrpc.c \
daemon/lightningd.c \ daemon/lightningd.c \
daemon/peer.c \
daemon/timeout.c daemon/timeout.c
DAEMON_OBJS := $(DAEMON_SRC:.c=.o) DAEMON_OBJS := $(DAEMON_SRC:.c=.o)
@ -31,6 +32,7 @@ DAEMON_HEADERS := \
daemon/jsonrpc.h \ daemon/jsonrpc.h \
daemon/lightningd.h \ daemon/lightningd.h \
daemon/log.h \ daemon/log.h \
daemon/peer.h \
daemon/pseudorand.h \ daemon/pseudorand.h \
daemon/timeout.h daemon/timeout.h

9
daemon/lightningd.c

@ -2,6 +2,7 @@
#include "jsonrpc.h" #include "jsonrpc.h"
#include "lightningd.h" #include "lightningd.h"
#include "log.h" #include "log.h"
#include "peer.h"
#include "timeout.h" #include "timeout.h"
#include <ccan/container_of/container_of.h> #include <ccan/container_of/container_of.h>
#include <ccan/err/err.h> #include <ccan/err/err.h>
@ -12,6 +13,7 @@
#include <ccan/timer/timer.h> #include <ccan/timer/timer.h>
#include <errno.h> #include <errno.h>
#include <string.h> #include <string.h>
#include <sys/socket.h>
#include <sys/stat.h> #include <sys/stat.h>
#include <sys/types.h> #include <sys/types.h>
#include <unistd.h> #include <unistd.h>
@ -25,6 +27,7 @@ static struct lightningd_state *lightningd_state(void)
state->base_log = new_log(state, state->log_record, state->base_log = new_log(state, state->log_record,
"lightningd(%u):", (int)getpid()); "lightningd(%u):", (int)getpid());
list_head_init(&state->peers);
timers_init(&state->timers, time_now()); timers_init(&state->timers, time_now());
return state; return state;
} }
@ -52,6 +55,7 @@ int main(int argc, char *argv[])
{ {
struct lightningd_state *state = lightningd_state(); struct lightningd_state *state = lightningd_state();
struct timer *expired; struct timer *expired;
unsigned int portnum = 0;
err_set_progname(argv[0]); err_set_progname(argv[0]);
opt_set_alloc(opt_allocfn, tal_reallocfn, tal_freefn); opt_set_alloc(opt_allocfn, tal_reallocfn, tal_freefn);
@ -60,6 +64,8 @@ int main(int argc, char *argv[])
"\n" "\n"
"A bitcoin lightning daemon.", "A bitcoin lightning daemon.",
"Print this message."); "Print this message.");
opt_register_arg("--port", opt_set_uintval, NULL, &portnum,
"Port to bind to (otherwise, dynamic port is used)");
opt_register_logging(state->base_log); opt_register_logging(state->base_log);
opt_register_version(); opt_register_version();
@ -94,6 +100,9 @@ int main(int argc, char *argv[])
/* Create RPC socket (if any) */ /* Create RPC socket (if any) */
setup_jsonrpc(state, state->rpc_filename); setup_jsonrpc(state, state->rpc_filename);
/* Set up connections from peers. */
setup_listeners(state, portnum);
log_info(state->base_log, "Hello world!"); log_info(state->base_log, "Hello world!");
/* If io_loop returns NULL, either a timer expired, or all fds closed */ /* If io_loop returns NULL, either a timer expired, or all fds closed */

4
daemon/lightningd.h

@ -1,6 +1,7 @@
#ifndef LIGHTNING_DAEMON_LIGHTNING_H #ifndef LIGHTNING_DAEMON_LIGHTNING_H
#define LIGHTNING_DAEMON_LIGHTNING_H #define LIGHTNING_DAEMON_LIGHTNING_H
#include "config.h" #include "config.h"
#include <ccan/list/list.h>
#include <ccan/timer/timer.h> #include <ccan/timer/timer.h>
#include <stdio.h> #include <stdio.h>
@ -17,5 +18,8 @@ struct lightningd_state {
/* Any pending timers. */ /* Any pending timers. */
struct timers timers; struct timers timers;
/* Our peers. */
struct list_head peers;
}; };
#endif /* LIGHTNING_DAEMON_LIGHTNING_H */ #endif /* LIGHTNING_DAEMON_LIGHTNING_H */

178
daemon/peer.c

@ -0,0 +1,178 @@
#include "lightningd.h"
#include "log.h"
#include "peer.h"
#include <arpa/inet.h>
#include <ccan/io/io.h>
#include <ccan/noerr/noerr.h>
#include <ccan/short_types/short_types.h>
#include <ccan/tal/str/str.h>
#include <ccan/tal/tal.h>
#include <errno.h>
#include <netinet/in.h>
#include <stdlib.h>
#include <sys/socket.h>
#include <sys/types.h>
static u16 get_port(const struct netaddr *addr)
{
switch (addr->saddr.s.sa_family) {
case AF_INET:
return ntohs(addr->saddr.ipv4.sin_port);
case AF_INET6:
return ntohs(addr->saddr.ipv6.sin6_port);
default:
abort();
}
}
static void destroy_peer(struct peer *peer)
{
list_del_from(&peer->state->peers, &peer->list);
}
static struct peer *new_peer(struct lightningd_state *state,
struct io_conn *conn,
int addr_type, int addr_protocol,
const char *in_or_out)
{
struct peer *peer = tal(state, struct peer);
char name[INET6_ADDRSTRLEN];
/* FIXME: Stop listening if too many peers? */
list_add(&state->peers, &peer->list);
peer->state = state;
peer->addr.type = addr_type;
peer->addr.protocol = addr_protocol;
/* FIXME: Attach IO logging for this peer. */
tal_add_destructor(peer, destroy_peer);
peer->addr.addrlen = sizeof(peer->addr.saddr);
if (getpeername(io_conn_fd(conn), &peer->addr.saddr.s,
&peer->addr.addrlen) != 0) {
log_unusual(state->base_log,
"Could not get address for peer: %s",
strerror(errno));
return tal_free(peer);
}
if (!inet_ntop(peer->addr.saddr.s.sa_family, &peer->addr.saddr,
name, sizeof(name)))
strcpy(name, "UNCONVERTABLE-ADDR");
peer->log = new_log(peer, state->log_record, "%s-%s:%s:%u",
log_prefix(state->base_log), in_or_out,
name, get_port(&peer->addr));
return peer;
}
struct io_plan *peer_connected_out(struct io_conn *conn,
struct lightningd_state *state,
const char *name, const char *port)
{
struct peer *peer = new_peer(state, conn, SOCK_STREAM, IPPROTO_TCP,
"out");
if (!peer) {
log_unusual(peer->log, "Failed to make peer for %s:%s",
name, port);
return io_close(conn);
}
log_info(peer->log, "Connected out to %s:%s", name, port);
return io_write(conn, "Hello!", 6, io_close_cb, NULL);
}
static struct io_plan *peer_connected_in(struct io_conn *conn,
struct lightningd_state *state)
{
struct peer *peer = new_peer(state, conn, SOCK_STREAM, IPPROTO_TCP,
"in");
if (!peer)
return io_close(conn);
return io_write(conn, "Hello!", 6, io_close_cb, NULL);
}
static int make_listen_fd(struct lightningd_state *state,
int domain, void *addr, socklen_t len)
{
int fd = socket(domain, SOCK_STREAM, 0);
if (fd < 0) {
log_debug(state->base_log, "Failed to create %u socket: %s",
domain, strerror(errno));
return -1;
}
if (!addr || bind(fd, addr, len) == 0) {
if (listen(fd, 5) == 0)
return fd;
log_unusual(state->base_log, "Failed to listen on %u socket: %s",
domain, strerror(errno));
} else
log_debug(state->base_log, "Failed to bind on %u socket: %s",
domain, strerror(errno));
close_noerr(fd);
return -1;
}
void setup_listeners(struct lightningd_state *state, unsigned int portnum)
{
struct sockaddr_in addr;
struct sockaddr_in6 addr6;
socklen_t len;
int fd1, fd2;
u16 listen_port;
addr.sin_family = AF_INET;
addr.sin_addr.s_addr = INADDR_ANY;
addr.sin_port = htons(portnum);
addr6.sin6_family = AF_INET6;
addr6.sin6_addr = in6addr_any;
addr6.sin6_port = htons(portnum);
/* IPv6, since on Linux that (usually) binds to IPv4 too. */
fd1 = make_listen_fd(state, AF_INET6, portnum ? &addr6 : NULL,
sizeof(addr6));
if (fd1 >= 0) {
struct sockaddr_in6 in6;
len = sizeof(in6);
if (getsockname(fd1, (void *)&in6, &len) != 0) {
log_unusual(state->base_log,
"Failed get IPv6 sockname: %s",
strerror(errno));
close_noerr(fd1);
} else {
addr.sin_port = in6.sin6_port;
listen_port = ntohs(addr.sin_port);
log_info(state->base_log,
"Creating IPv6 listener on port %u",
listen_port);
io_new_listener(state, fd1, peer_connected_in, state);
}
}
/* Just in case, aim for the same port... */
fd2 = make_listen_fd(state, AF_INET,
addr.sin_port ? &addr : NULL, sizeof(addr));
if (fd2 >= 0) {
len = sizeof(addr);
if (getsockname(fd2, (void *)&addr, &len) != 0) {
log_unusual(state->base_log,
"Failed get IPv4 sockname: %s",
strerror(errno));
close_noerr(fd2);
} else {
listen_port = ntohs(addr.sin_port);
log_info(state->base_log,
"Creating IPv4 listener on port %u",
listen_port);
io_new_listener(state, fd2, peer_connected_in, state);
}
}
if (fd1 < 0 && fd2 < 0)
fatal("Could not bind to a network address");
}

43
daemon/peer.h

@ -0,0 +1,43 @@
#ifndef LIGHTNING_DAEMON_PEER_H
#define LIGHTNING_DAEMON_PEER_H
#include "config.h"
#include <ccan/list/list.h>
#include <netinet/in.h>
#include <netinet/ip.h>
#include <sys/socket.h>
#include <sys/socket.h>
/* This can be extended to support other protocols in future. */
struct netaddr {
int type; /* See socket(2): SOCK_STREAM currently */
int protocol; /* See socket(2): 0 currently */
socklen_t addrlen;
union {
struct sockaddr s;
struct sockaddr_in ipv4;
struct sockaddr_in6 ipv6;
} saddr;
};
struct peer {
/* state->peers list */
struct list_node list;
/* Global state. */
struct lightningd_state *state;
/* The other end's address. */
struct netaddr addr;
/* What happened. */
struct log *log;
};
struct io_conn;
struct io_plan *peer_connected_out(struct io_conn *conn,
struct lightningd_state *state,
const char *name, const char *port);
void setup_listeners(struct lightningd_state *state, unsigned int portnum);
#endif /* LIGHTNING_DAEMON_PEER_H */
Loading…
Cancel
Save