@ -1,10 +1,10 @@
# include <arpa/inet.h>
# include <arpa/inet.h>
# include <assert.h>
# include <assert.h>
# include <common/status.h>
# include <common/type_to_string.h>
# include <common/wireaddr.h>
# include <common/wireaddr.h>
# include <errno.h>
# include <errno.h>
# include <lightningd/lightningd.h>
# include <gossipd/netaddress.h>
# include <lightningd/log.h>
# include <lightningd/netaddress.h>
# include <netinet/in.h>
# include <netinet/in.h>
# include <stdbool.h>
# include <stdbool.h>
# include <sys/socket.h>
# include <sys/socket.h>
@ -202,26 +202,25 @@ static bool IsRoutable(const struct wireaddr *addr)
* then query address . */
* then query address . */
/* Returns 0 if protocol completely unsupported, ADDR_LISTEN if we
/* Returns 0 if protocol completely unsupported, ADDR_LISTEN if we
* can ' t reach addr , ADDR_LISTEN_AND_ANNOUNCE if we can ( and fill saddr ) . */
* can ' t reach addr , ADDR_LISTEN_AND_ANNOUNCE if we can ( and fill saddr ) . */
static enum addr_listen_announce get_local_sockname ( struct lightningd * ld ,
static enum addr_listen_announce get_local_sockname ( int af , void * saddr ,
int af , void * saddr ,
socklen_t saddrlen )
socklen_t saddrlen )
{
{
int fd = socket ( af , SOCK_DGRAM , 0 ) ;
int fd = socket ( af , SOCK_DGRAM , 0 ) ;
if ( fd < 0 ) {
if ( fd < 0 ) {
log_debug ( ld - > log , " Failed to create %u socket: %s " ,
status_trace ( " Failed to create %u socket: %s " ,
af , strerror ( errno ) ) ;
af , strerror ( errno ) ) ;
return 0 ;
return 0 ;
}
}
if ( connect ( fd , saddr , saddrlen ) ! = 0 ) {
if ( connect ( fd , saddr , saddrlen ) ! = 0 ) {
log_debug ( ld - > log , " Failed to connect %u socket: %s " ,
status_trace ( " Failed to connect %u socket: %s " ,
af , strerror ( errno ) ) ;
af , strerror ( errno ) ) ;
close ( fd ) ;
close ( fd ) ;
return ADDR_LISTEN ;
return ADDR_LISTEN ;
}
}
if ( getsockname ( fd , saddr , & saddrlen ) ! = 0 ) {
if ( getsockname ( fd , saddr , & saddrlen ) ! = 0 ) {
log_debug ( ld - > log , " Failed to get %u socket name: %s " ,
status_trace ( " Failed to get %u socket name: %s " ,
af , strerror ( errno ) ) ;
af , strerror ( errno ) ) ;
close ( fd ) ;
close ( fd ) ;
return ADDR_LISTEN ;
return ADDR_LISTEN ;
@ -234,8 +233,7 @@ static enum addr_listen_announce get_local_sockname(struct lightningd *ld,
/* Return 0 if not available, or whether it's listenable-only or announceable.
/* Return 0 if not available, or whether it's listenable-only or announceable.
* If it ' s listenable only , will set wireaddr to all - zero address for universal
* If it ' s listenable only , will set wireaddr to all - zero address for universal
* binding . */
* binding . */
static enum addr_listen_announce guess_one_address ( struct lightningd * ld ,
static enum addr_listen_announce guess_one_address ( struct wireaddr * addr ,
struct wireaddr * addr ,
u16 portnum ,
u16 portnum ,
enum wire_addr_type type )
enum wire_addr_type type )
{
{
@ -252,7 +250,7 @@ static enum addr_listen_announce guess_one_address(struct lightningd *ld,
/* 8.8.8.8 */
/* 8.8.8.8 */
sin . sin_addr . s_addr = 0x08080808 ;
sin . sin_addr . s_addr = 0x08080808 ;
sin . sin_family = AF_INET ;
sin . sin_family = AF_INET ;
ret = get_local_sockname ( ld , AF_INET , & sin , sizeof ( sin ) ) ;
ret = get_local_sockname ( AF_INET , & sin , sizeof ( sin ) ) ;
addr - > addrlen = sizeof ( sin . sin_addr ) ;
addr - > addrlen = sizeof ( sin . sin_addr ) ;
memcpy ( addr - > addr , & sin . sin_addr , addr - > addrlen ) ;
memcpy ( addr - > addr , & sin . sin_addr , addr - > addrlen ) ;
break ;
break ;
@ -266,13 +264,13 @@ static enum addr_listen_announce guess_one_address(struct lightningd *ld,
sin6 . sin6_port = htons ( 53 ) ;
sin6 . sin6_port = htons ( 53 ) ;
sin6 . sin6_family = AF_INET6 ;
sin6 . sin6_family = AF_INET6 ;
memcpy ( sin6 . sin6_addr . s6_addr , pchGoogle , sizeof ( pchGoogle ) ) ;
memcpy ( sin6 . sin6_addr . s6_addr , pchGoogle , sizeof ( pchGoogle ) ) ;
ret = get_local_sockname ( ld , AF_INET6 , & sin6 , sizeof ( sin6 ) ) ;
ret = get_local_sockname ( AF_INET6 , & sin6 , sizeof ( sin6 ) ) ;
addr - > addrlen = sizeof ( sin6 . sin6_addr ) ;
addr - > addrlen = sizeof ( sin6 . sin6_addr ) ;
memcpy ( addr - > addr , & sin6 . sin6_addr , addr - > addrlen ) ;
memcpy ( addr - > addr , & sin6 . sin6_addr , addr - > addrlen ) ;
break ;
break ;
}
}
case ADDR_TYPE_PADDING :
case ADDR_TYPE_PADDING :
log_debug ( ld - > log , " Padding address, ignoring " ) ;
status_trace ( " Padding address, ignoring " ) ;
return 0 ;
return 0 ;
}
}
@ -281,7 +279,7 @@ static enum addr_listen_announce guess_one_address(struct lightningd *ld,
/* If we can reach it, but resulting address is unroutable, listen only */
/* If we can reach it, but resulting address is unroutable, listen only */
if ( ret = = ADDR_LISTEN_AND_ANNOUNCE & & ! IsRoutable ( addr ) ) {
if ( ret = = ADDR_LISTEN_AND_ANNOUNCE & & ! IsRoutable ( addr ) ) {
log_debug ( ld - > log , " Address %s is not routable " ,
status_trace ( " Address %s is not routable " ,
type_to_string ( tmpctx , struct wireaddr , addr ) ) ;
type_to_string ( tmpctx , struct wireaddr , addr ) ) ;
ret = ADDR_LISTEN ;
ret = ADDR_LISTEN ;
}
}
@ -290,33 +288,35 @@ static enum addr_listen_announce guess_one_address(struct lightningd *ld,
/* This corresponds to INADDR_ANY or in6addr_any */
/* This corresponds to INADDR_ANY or in6addr_any */
memset ( addr - > addr , 0 , addr - > addrlen ) ;
memset ( addr - > addr , 0 , addr - > addrlen ) ;
} else {
} else {
log_debug ( ld - > log , " Public address %s " ,
status_trace ( " Public address %s " ,
type_to_string ( tmpctx , struct wireaddr , addr ) ) ;
type_to_string ( tmpctx , struct wireaddr , addr ) ) ;
}
}
return ret ;
return ret ;
}
}
void guess_addresses ( struct lightningd * ld )
void guess_addresses ( struct wireaddr * * wireaddrs ,
enum addr_listen_announce * * listen_announce ,
u16 portnum )
{
{
size_t n = tal_count ( ld - > wireaddrs ) ;
size_t n = tal_count ( * wireaddrs ) ;
log_debug ( ld - > log , " Trying to guess public addresses... " ) ;
status_trace ( " Trying to guess public addresses... " ) ;
/* We allocate an extra, then remove if it's not needed. */
/* We allocate an extra, then remove if it's not needed. */
tal_resize ( & ld - > wireaddrs , n + 1 ) ;
tal_resize ( wireaddrs , n + 1 ) ;
tal_resize ( & ld - > listen_announce , n + 1 ) ;
tal_resize ( listen_announce , n + 1 ) ;
/* We do IPv6 first: on Linux, that binds to IPv4 too. */
/* We do IPv6 first: on Linux, that binds to IPv4 too. */
ld - > listen_announce [ n ] = guess_one_address ( ld , & ld - > wireaddrs [ n ] ,
( * listen_announce ) [ n ] = guess_one_address ( & ( * wireaddrs ) [ n ] ,
ld - > portnum , ADDR_TYPE_IPV6 ) ;
portnum , ADDR_TYPE_IPV6 ) ;
if ( ld - > listen_announce [ n ] ! = 0 ) {
if ( ( * listen_announce ) [ n ] ! = 0 ) {
n + + ;
n + + ;
tal_resize ( & ld - > wireaddrs , n + 1 ) ;
tal_resize ( wireaddrs , n + 1 ) ;
tal_resize ( & ld - > listen_announce , n + 1 ) ;
tal_resize ( listen_announce , n + 1 ) ;
}
}
ld - > listen_announce [ n ] = guess_one_address ( ld , & ld - > wireaddrs [ n ] ,
( * listen_announce ) [ n ] = guess_one_address ( & ( * wireaddrs ) [ n ] ,
ld - > portnum , ADDR_TYPE_IPV4 ) ;
portnum , ADDR_TYPE_IPV4 ) ;
if ( ld - > listen_announce [ n ] = = 0 ) {
if ( ( * listen_announce ) [ n ] = = 0 ) {
tal_resize ( & ld - > wireaddrs , n ) ;
tal_resize ( wireaddrs , n ) ;
tal_resize ( & ld - > listen_announce , n ) ;
tal_resize ( listen_announce , n ) ;
}
}
}
}