2008-02-03 07:07:59 -05:00
|
|
|
/*
|
|
|
|
* libretroshare/src/util: rsnet.cc
|
|
|
|
*
|
|
|
|
* Universal Networking Header for RetroShare.
|
|
|
|
*
|
|
|
|
* Copyright 2007-2008 by Robert Fernie.
|
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Library General Public
|
|
|
|
* License Version 2 as published by the Free Software Foundation.
|
|
|
|
*
|
|
|
|
* This library is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
* Library General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Library General Public
|
|
|
|
* License along with this library; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
|
|
|
|
* USA.
|
|
|
|
*
|
|
|
|
* Please report all bugs and problems to "retroshare@lunamutt.com".
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "util/rsnet.h"
|
2010-07-10 16:34:03 -04:00
|
|
|
#include "util/rsthreads.h"
|
2012-04-15 10:37:44 -04:00
|
|
|
#include "util/rsstring.h"
|
2015-10-07 21:24:31 -04:00
|
|
|
#include "util/rsmemory.h"
|
2010-04-22 16:03:47 -04:00
|
|
|
|
|
|
|
#ifdef WINDOWS_SYS
|
|
|
|
#else
|
2010-04-22 14:49:27 -04:00
|
|
|
#include <netdb.h>
|
2010-04-22 16:03:47 -04:00
|
|
|
#endif
|
2008-02-07 11:18:34 -05:00
|
|
|
|
2008-03-03 09:41:15 -05:00
|
|
|
/* enforce LITTLE_ENDIAN on Windows */
|
|
|
|
#ifdef WINDOWS_SYS
|
|
|
|
#define BYTE_ORDER 1234
|
|
|
|
#define LITTLE_ENDIAN 1234
|
|
|
|
#define BIG_ENDIAN 4321
|
|
|
|
#endif
|
2008-02-03 07:07:59 -05:00
|
|
|
|
2015-12-21 11:25:02 -05:00
|
|
|
#ifndef ntohll
|
2008-02-03 07:07:59 -05:00
|
|
|
uint64_t ntohll(uint64_t x)
|
|
|
|
{
|
|
|
|
#ifdef BYTE_ORDER
|
|
|
|
#if BYTE_ORDER == BIG_ENDIAN
|
|
|
|
return x;
|
|
|
|
#elif BYTE_ORDER == LITTLE_ENDIAN
|
|
|
|
|
|
|
|
uint32_t top = (uint32_t) (x >> 32);
|
|
|
|
uint32_t bot = (uint32_t) (0x00000000ffffffffULL & x);
|
|
|
|
|
|
|
|
uint64_t rev = ((uint64_t) ntohl(top)) | (((uint64_t) ntohl(bot)) << 32);
|
|
|
|
|
|
|
|
return rev;
|
|
|
|
#else
|
|
|
|
#error "ENDIAN determination Failed"
|
|
|
|
#endif
|
|
|
|
#else
|
|
|
|
#error "ENDIAN determination Failed (BYTE_ORDER not defined)"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
}
|
2015-12-21 11:25:02 -05:00
|
|
|
#endif
|
|
|
|
#ifndef htonll
|
2008-02-03 07:07:59 -05:00
|
|
|
uint64_t htonll(uint64_t x)
|
|
|
|
{
|
|
|
|
return ntohll(x);
|
|
|
|
}
|
2015-12-21 11:25:02 -05:00
|
|
|
#endif
|
2008-02-03 07:07:59 -05:00
|
|
|
|
2008-02-07 11:18:34 -05:00
|
|
|
void sockaddr_clear(struct sockaddr_in *addr)
|
|
|
|
{
|
|
|
|
memset(addr, 0, sizeof(struct sockaddr_in));
|
|
|
|
addr->sin_family = AF_INET;
|
|
|
|
}
|
2008-02-03 07:07:59 -05:00
|
|
|
|
2015-10-07 21:24:31 -04:00
|
|
|
bool rsGetHostByName(const std::string& hostname, in_addr& returned_addr)
|
|
|
|
{
|
2016-05-09 20:54:58 -04:00
|
|
|
addrinfo *info = NULL;
|
|
|
|
int res = getaddrinfo(hostname.c_str(),NULL,NULL,&info) ;
|
|
|
|
|
2017-04-22 04:13:23 -04:00
|
|
|
bool ok = true;
|
2016-05-10 09:46:33 -04:00
|
|
|
if(res > 0 || info == NULL || info->ai_addr == NULL)
|
2016-05-09 20:54:58 -04:00
|
|
|
{
|
|
|
|
std::cerr << "(EE) getaddrinfo returned error " << res << " on string \"" << hostname << "\"" << std::endl;
|
|
|
|
returned_addr.s_addr = 0 ;
|
2017-04-22 04:13:23 -04:00
|
|
|
ok = false;
|
2016-05-09 20:54:58 -04:00
|
|
|
}
|
|
|
|
else
|
2016-05-10 09:46:33 -04:00
|
|
|
returned_addr.s_addr = ((sockaddr_in*)info->ai_addr)->sin_addr.s_addr ;
|
2016-05-09 20:54:58 -04:00
|
|
|
|
2017-04-22 04:13:23 -04:00
|
|
|
if(info)
|
|
|
|
freeaddrinfo(info) ;
|
2016-05-09 20:54:58 -04:00
|
|
|
|
|
|
|
#ifdef DEPRECATED_TO_REMOVE
|
2015-10-20 12:41:40 -04:00
|
|
|
#if defined(WINDOWS_SYS) || defined(__APPLE__) || defined(__HAIKU__)
|
2015-10-08 18:54:12 -04:00
|
|
|
hostent *result = gethostbyname(hostname.c_str()) ;
|
|
|
|
#else
|
2015-10-07 21:24:31 -04:00
|
|
|
RsTemporaryMemory mem(8192) ;
|
|
|
|
|
|
|
|
if(!mem)
|
|
|
|
{
|
|
|
|
std::cerr << __PRETTY_FUNCTION__ << ": Cannot allocate memory!" << std::endl;
|
|
|
|
return false; // Do something.
|
|
|
|
}
|
|
|
|
|
|
|
|
int error = 0;
|
|
|
|
struct hostent pHost;
|
|
|
|
struct hostent *result;
|
|
|
|
|
|
|
|
if(gethostbyname_r(hostname.c_str(), &pHost, (char*)(unsigned char*)mem, mem.size(), &result, &error) != 0)
|
|
|
|
{
|
|
|
|
std::cerr << __PRETTY_FUNCTION__ << ": cannot call gethostname_r. Internal error reported. Check buffer size." << std::endl;
|
|
|
|
return false ;
|
|
|
|
}
|
2015-10-08 18:54:12 -04:00
|
|
|
#endif
|
2015-10-07 21:24:31 -04:00
|
|
|
if(!result)
|
|
|
|
{
|
2015-10-08 18:54:12 -04:00
|
|
|
std::cerr << __PRETTY_FUNCTION__ << ": gethostname returned null result." << std::endl;
|
2015-10-07 21:24:31 -04:00
|
|
|
return false ;
|
|
|
|
}
|
|
|
|
// Use contents of result.
|
|
|
|
|
|
|
|
returned_addr.s_addr = *(unsigned long*) (result->h_addr);
|
2016-05-09 20:54:58 -04:00
|
|
|
#endif
|
2015-10-07 21:24:31 -04:00
|
|
|
|
2017-04-22 04:13:23 -04:00
|
|
|
return ok;
|
2015-10-07 21:24:31 -04:00
|
|
|
}
|
2008-06-07 07:19:09 -04:00
|
|
|
|
2010-05-23 18:27:10 -04:00
|
|
|
bool isValidNet(const struct in_addr *addr)
|
2008-06-07 07:19:09 -04:00
|
|
|
{
|
|
|
|
// invalid address.
|
|
|
|
if((*addr).s_addr == INADDR_NONE)
|
|
|
|
return false;
|
|
|
|
if((*addr).s_addr == 0)
|
|
|
|
return false;
|
|
|
|
// should do more tests.
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-05-23 18:27:10 -04:00
|
|
|
bool isLoopbackNet(const struct in_addr *addr)
|
2008-06-07 07:19:09 -04:00
|
|
|
{
|
|
|
|
in_addr_t taddr = ntohl(addr->s_addr);
|
|
|
|
return (taddr == (127 << 24 | 1));
|
|
|
|
}
|
|
|
|
|
2010-05-23 18:27:10 -04:00
|
|
|
bool isPrivateNet(const struct in_addr *addr)
|
2008-06-07 07:19:09 -04:00
|
|
|
{
|
|
|
|
in_addr_t taddr = ntohl(addr->s_addr);
|
|
|
|
|
|
|
|
// 10.0.0.0/8
|
|
|
|
// 172.16.0.0/12
|
|
|
|
// 192.168.0.0/16
|
|
|
|
// 169.254.0.0/16
|
|
|
|
if ((taddr>>24 == 10) ||
|
|
|
|
(taddr>>20 == (172<<4 | 16>>4)) ||
|
|
|
|
(taddr>>16 == (192<<8 | 168)) ||
|
|
|
|
(taddr>>16 == (169<<8 | 254)))
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-05-23 18:27:10 -04:00
|
|
|
bool isExternalNet(const struct in_addr *addr)
|
2008-06-07 07:19:09 -04:00
|
|
|
{
|
|
|
|
if (!isValidNet(addr))
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (isLoopbackNet(addr))
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (isPrivateNet(addr))
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2010-06-10 09:42:49 -04:00
|
|
|
std::ostream &operator<<(std::ostream &out, const struct sockaddr_in &addr)
|
|
|
|
{
|
2012-03-17 21:33:25 -04:00
|
|
|
out << "[" << rs_inet_ntoa(addr.sin_addr) << ":";
|
2010-06-10 09:42:49 -04:00
|
|
|
out << htons(addr.sin_port) << "]";
|
|
|
|
return out;
|
|
|
|
}
|
2010-07-10 16:34:03 -04:00
|
|
|
|
|
|
|
/* thread-safe version of inet_ntoa */
|
|
|
|
|
|
|
|
std::string rs_inet_ntoa(struct in_addr in)
|
|
|
|
{
|
2012-04-15 10:37:44 -04:00
|
|
|
std::string str;
|
2011-06-29 14:02:44 -04:00
|
|
|
uint8_t *bytes = (uint8_t *) &(in.s_addr);
|
2012-04-15 10:37:44 -04:00
|
|
|
rs_sprintf(str, "%u.%u.%u.%u", (int) bytes[0], (int) bytes[1], (int) bytes[2], (int) bytes[3]);
|
|
|
|
return str;
|
2010-07-10 16:34:03 -04:00
|
|
|
}
|