239 lines
5.9 KiB
C
239 lines
5.9 KiB
C
/* $Id: upnputils.c,v 1.12 2018/03/13 10:25:20 nanard Exp $ */
|
|
/* MiniUPnP project
|
|
* http://miniupnp.free.fr/ or http://miniupnp.tuxfamily.org/
|
|
* (c) 2006-2018 Thomas Bernard
|
|
* This software is subject to the conditions detailed
|
|
* in the LICENCE file provided within the distribution */
|
|
|
|
#include "config.h"
|
|
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <syslog.h>
|
|
#include <unistd.h>
|
|
#include <fcntl.h>
|
|
#include <time.h>
|
|
#include <sys/types.h>
|
|
#include <sys/socket.h>
|
|
#include <netinet/in.h>
|
|
#include <arpa/inet.h>
|
|
#ifdef AF_LINK
|
|
#include <net/if_dl.h>
|
|
#endif
|
|
#include <errno.h>
|
|
|
|
#include "upnputils.h"
|
|
#include "upnpglobalvars.h"
|
|
#ifdef ENABLE_IPV6
|
|
#include "getroute.h"
|
|
#endif
|
|
|
|
int
|
|
sockaddr_to_string(const struct sockaddr * addr, char * str, size_t size)
|
|
{
|
|
char buffer[64];
|
|
unsigned short port = 0;
|
|
int n = -1;
|
|
|
|
switch(addr->sa_family)
|
|
{
|
|
#ifdef AF_INET6
|
|
case AF_INET6:
|
|
if(inet_ntop(addr->sa_family,
|
|
&((struct sockaddr_in6 *)addr)->sin6_addr,
|
|
buffer, sizeof(buffer)) == NULL) {
|
|
snprintf(buffer, sizeof(buffer), "inet_ntop: %s", strerror(errno));
|
|
}
|
|
port = ntohs(((struct sockaddr_in6 *)addr)->sin6_port);
|
|
if(((struct sockaddr_in6 *)addr)->sin6_scope_id > 0) {
|
|
char ifname[IF_NAMESIZE];
|
|
if(if_indextoname(((struct sockaddr_in6 *)addr)->sin6_scope_id, ifname) == NULL)
|
|
strncpy(ifname, "ERROR", sizeof(ifname));
|
|
n = snprintf(str, size, "[%s%%%s]:%hu", buffer, ifname, port);
|
|
} else {
|
|
n = snprintf(str, size, "[%s]:%hu", buffer, port);
|
|
}
|
|
break;
|
|
#endif /* AF_INET6 */
|
|
case AF_INET:
|
|
if(inet_ntop(addr->sa_family,
|
|
&((struct sockaddr_in *)addr)->sin_addr,
|
|
buffer, sizeof(buffer)) == NULL) {
|
|
snprintf(buffer, sizeof(buffer), "inet_ntop: %s", strerror(errno));
|
|
}
|
|
port = ntohs(((struct sockaddr_in *)addr)->sin_port);
|
|
n = snprintf(str, size, "%s:%hu", buffer, port);
|
|
break;
|
|
#ifdef AF_LINK
|
|
#if defined(__sun)
|
|
/* solaris does not seem to have link_ntoa */
|
|
/* #define link_ntoa _link_ntoa */
|
|
#define link_ntoa(x) "dummy-link_ntoa"
|
|
#endif
|
|
case AF_LINK:
|
|
{
|
|
struct sockaddr_dl * sdl = (struct sockaddr_dl *)addr;
|
|
n = snprintf(str, size, "index=%hu type=%d %s",
|
|
sdl->sdl_index, sdl->sdl_type,
|
|
link_ntoa(sdl));
|
|
}
|
|
break;
|
|
#endif /* AF_LINK */
|
|
default:
|
|
n = snprintf(str, size, "unknown address family %d", addr->sa_family);
|
|
#if 0
|
|
n = snprintf(str, size, "unknown address family %d "
|
|
"%02x %02x %02x %02x %02x %02x %02x %02x",
|
|
addr->sa_family,
|
|
addr->sa_data[0], addr->sa_data[1], (unsigned)addr->sa_data[2], addr->sa_data[3],
|
|
addr->sa_data[4], addr->sa_data[5], (unsigned)addr->sa_data[6], addr->sa_data[7]);
|
|
#endif
|
|
}
|
|
return n;
|
|
}
|
|
|
|
|
|
int
|
|
set_non_blocking(int fd)
|
|
{
|
|
int flags = fcntl(fd, F_GETFL);
|
|
if(flags < 0)
|
|
return 0;
|
|
if(fcntl(fd, F_SETFL, flags | O_NONBLOCK) < 0)
|
|
return 0;
|
|
return 1;
|
|
}
|
|
|
|
struct lan_addr_s *
|
|
get_lan_for_peer(const struct sockaddr * peer)
|
|
{
|
|
struct lan_addr_s * lan_addr = NULL;
|
|
#ifdef DEBUG
|
|
char dbg_str[64];
|
|
#endif /* DEBUG */
|
|
|
|
#ifdef ENABLE_IPV6
|
|
if(peer->sa_family == AF_INET6)
|
|
{
|
|
struct sockaddr_in6 * peer6 = (struct sockaddr_in6 *)peer;
|
|
if(IN6_IS_ADDR_V4MAPPED(&peer6->sin6_addr))
|
|
{
|
|
struct in_addr peer_addr;
|
|
memcpy(&peer_addr, &peer6->sin6_addr.s6_addr[12], 4);
|
|
for(lan_addr = lan_addrs.lh_first;
|
|
lan_addr != NULL;
|
|
lan_addr = lan_addr->list.le_next)
|
|
{
|
|
if( (peer_addr.s_addr & lan_addr->mask.s_addr)
|
|
== (lan_addr->addr.s_addr & lan_addr->mask.s_addr))
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
int index = -1;
|
|
if(peer6->sin6_scope_id > 0)
|
|
index = (int)peer6->sin6_scope_id;
|
|
else
|
|
{
|
|
if(get_src_for_route_to(peer, NULL, NULL, &index) < 0)
|
|
return NULL;
|
|
}
|
|
syslog(LOG_DEBUG, "%s looking for LAN interface index=%d",
|
|
"get_lan_for_peer()", index);
|
|
for(lan_addr = lan_addrs.lh_first;
|
|
lan_addr != NULL;
|
|
lan_addr = lan_addr->list.le_next)
|
|
{
|
|
syslog(LOG_DEBUG,
|
|
"ifname=%s index=%u str=%s addr=%08x mask=%08x",
|
|
lan_addr->ifname, lan_addr->index,
|
|
lan_addr->str,
|
|
ntohl(lan_addr->addr.s_addr),
|
|
ntohl(lan_addr->mask.s_addr));
|
|
if(index == (int)lan_addr->index)
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
else if(peer->sa_family == AF_INET)
|
|
{
|
|
#endif /* ENABLE_IPV6 */
|
|
for(lan_addr = lan_addrs.lh_first;
|
|
lan_addr != NULL;
|
|
lan_addr = lan_addr->list.le_next)
|
|
{
|
|
if( (((const struct sockaddr_in *)peer)->sin_addr.s_addr & lan_addr->mask.s_addr)
|
|
== (lan_addr->addr.s_addr & lan_addr->mask.s_addr))
|
|
break;
|
|
}
|
|
#ifdef ENABLE_IPV6
|
|
}
|
|
#endif /* ENABLE_IPV6 */
|
|
|
|
#ifdef DEBUG
|
|
sockaddr_to_string(peer, dbg_str, sizeof(dbg_str));
|
|
if(lan_addr) {
|
|
syslog(LOG_DEBUG, "%s: %s found in LAN %s %s",
|
|
"get_lan_for_peer()", dbg_str,
|
|
lan_addr->ifname, lan_addr->str);
|
|
} else {
|
|
syslog(LOG_DEBUG, "%s: %s not found !", "get_lan_for_peer()",
|
|
dbg_str);
|
|
}
|
|
#endif /* DEBUG */
|
|
return lan_addr;
|
|
}
|
|
|
|
time_t upnp_time(void)
|
|
{
|
|
#if defined(CLOCK_MONOTONIC_FAST) || defined(CLOCK_MONOTONIC)
|
|
#if defined(CLOCK_MONOTONIC_FAST)
|
|
#define UPNP_CLOCKID CLOCK_MONOTONIC_FAST
|
|
#else
|
|
#define UPNP_CLOCKID CLOCK_MONOTONIC
|
|
#endif
|
|
struct timespec ts;
|
|
if (clock_gettime(UPNP_CLOCKID, &ts) < 0)
|
|
return time(NULL);
|
|
else
|
|
return ts.tv_sec;
|
|
#else
|
|
return time(NULL);
|
|
#endif
|
|
}
|
|
|
|
time_t upnp_get_uptime(void)
|
|
{
|
|
#if defined(CLOCK_UPTIME_FAST) || defined(CLOCK_UPTIME)
|
|
#if defined(CLOCK_UPTIME_FAST)
|
|
#define UPNP_CLOCKID_UPTIME CLOCK_UPTIME_FAST
|
|
#else
|
|
#define UPNP_CLOCKID_UPTIME CLOCK_UPTIME
|
|
#endif
|
|
if(GETFLAG(SYSUPTIMEMASK))
|
|
{
|
|
struct timespec ts;
|
|
if (clock_gettime(UPNP_CLOCKID_UPTIME, &ts) >= 0)
|
|
return ts.tv_sec;
|
|
}
|
|
#endif
|
|
return upnp_time() - startup_time;
|
|
}
|
|
|
|
int upnp_gettimeofday(struct timeval * tv)
|
|
{
|
|
#if defined(CLOCK_MONOTONIC_FAST) || defined(CLOCK_MONOTONIC)
|
|
struct timespec ts;
|
|
int ret_code = clock_gettime(UPNP_CLOCKID, &ts);
|
|
if (ret_code == 0)
|
|
{
|
|
tv->tv_sec = ts.tv_sec;
|
|
tv->tv_usec = ts.tv_nsec / 1000;
|
|
}
|
|
return ret_code;
|
|
#else
|
|
return gettimeofday(tv, NULL);
|
|
#endif
|
|
}
|