2013-06-08 18:59:11 -07:00
|
|
|
#include "net/resolve.h"
|
|
|
|
#include "base/logging.h"
|
2014-07-13 08:15:14 +10:00
|
|
|
#include "base/timeutil.h"
|
2012-06-03 19:01:08 +02:00
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
2013-06-08 18:59:11 -07:00
|
|
|
#include <string>
|
2012-06-03 19:01:08 +02:00
|
|
|
|
|
|
|
|
2013-06-09 22:14:19 +10:00
|
|
|
#ifdef _WIN32
|
2012-06-03 19:01:08 +02:00
|
|
|
#include <WinSock2.h>
|
|
|
|
#include <Ws2tcpip.h>
|
2012-08-08 00:10:34 +02:00
|
|
|
#undef min
|
|
|
|
#undef max
|
2013-06-09 22:14:19 +10:00
|
|
|
#else
|
|
|
|
#include <netinet/in.h>
|
2014-06-10 23:22:20 +10:00
|
|
|
#include <arpa/inet.h>
|
2013-06-09 22:14:19 +10:00
|
|
|
#include <netdb.h>
|
|
|
|
#include <sys/socket.h>
|
2013-06-09 22:42:34 +10:00
|
|
|
#include <unistd.h>
|
2012-06-03 19:01:08 +02:00
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
namespace net {
|
|
|
|
|
|
|
|
void Init()
|
|
|
|
{
|
|
|
|
#ifdef _WIN32
|
2013-05-31 23:04:42 +02:00
|
|
|
// WSA does its own internal reference counting, no need to keep track of if we inited or not.
|
2012-10-30 13:20:55 +01:00
|
|
|
WSADATA wsaData = {0};
|
|
|
|
WSAStartup(MAKEWORD(2, 2), &wsaData);
|
2012-06-03 19:01:08 +02:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2012-07-15 17:04:27 +02:00
|
|
|
void Shutdown()
|
|
|
|
{
|
|
|
|
#ifdef _WIN32
|
2012-10-30 13:20:55 +01:00
|
|
|
WSACleanup();
|
2012-07-15 17:04:27 +02:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2013-02-28 23:59:32 -08:00
|
|
|
char *DNSResolveTry(const char *host, const char **err)
|
2012-06-03 19:01:08 +02:00
|
|
|
{
|
2012-10-30 13:20:55 +01:00
|
|
|
struct hostent *hent;
|
|
|
|
if((hent = gethostbyname(host)) == NULL)
|
|
|
|
{
|
2013-02-28 23:59:32 -08:00
|
|
|
*err = "Can't get IP";
|
|
|
|
return NULL;
|
2012-10-30 13:20:55 +01:00
|
|
|
}
|
|
|
|
int iplen = 15; //XXX.XXX.XXX.XXX
|
|
|
|
char *ip = (char *)malloc(iplen+1);
|
|
|
|
memset(ip, 0, iplen+1);
|
2013-03-04 14:48:18 -08:00
|
|
|
char *iptoa = inet_ntoa(*(in_addr *)hent->h_addr_list[0]);
|
|
|
|
if (iptoa == NULL)
|
2012-10-30 13:20:55 +01:00
|
|
|
{
|
2013-02-28 23:59:32 -08:00
|
|
|
*err = "Can't resolve host";
|
2013-05-30 15:08:51 -04:00
|
|
|
free(ip);
|
2013-02-28 23:59:32 -08:00
|
|
|
return NULL;
|
|
|
|
}
|
2013-03-04 14:48:18 -08:00
|
|
|
strncpy(ip, iptoa, iplen);
|
2013-02-28 23:59:32 -08:00
|
|
|
return ip;
|
|
|
|
}
|
|
|
|
|
|
|
|
char *DNSResolve(const char *host)
|
|
|
|
{
|
|
|
|
const char *err;
|
|
|
|
char *ip = DNSResolveTry(host, &err);
|
|
|
|
if (ip == NULL)
|
|
|
|
{
|
|
|
|
perror(err);
|
2012-10-30 13:20:55 +01:00
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
return ip;
|
2012-06-03 19:01:08 +02:00
|
|
|
}
|
|
|
|
|
2013-06-08 18:59:11 -07:00
|
|
|
bool DNSResolve(const std::string &host, const std::string &service, addrinfo **res, std::string &error)
|
|
|
|
{
|
|
|
|
addrinfo hints = {0};
|
|
|
|
// TODO: Might be uses to lookup other values.
|
|
|
|
hints.ai_socktype = SOCK_STREAM;
|
2013-06-09 22:14:19 +10:00
|
|
|
#ifdef BLACKBERRY
|
|
|
|
hints.ai_flags = 0;
|
2013-06-10 23:20:45 +02:00
|
|
|
#elif ANDROID
|
|
|
|
hints.ai_flags = AI_ADDRCONFIG;
|
2013-06-09 22:14:19 +10:00
|
|
|
#else
|
2013-06-10 23:20:45 +02:00
|
|
|
// AI_V4MAPPED seems to have issues on some platforms, not sure we should include it:
|
|
|
|
// http://stackoverflow.com/questions/1408030/what-is-the-purpose-of-the-ai-v4mapped-flag-in-getaddrinfo
|
2013-06-11 09:30:19 +02:00
|
|
|
hints.ai_flags = /*AI_V4MAPPED |*/ AI_ADDRCONFIG;
|
2013-06-09 22:14:19 +10:00
|
|
|
#endif
|
2013-06-08 18:59:11 -07:00
|
|
|
hints.ai_protocol = IPPROTO_TCP;
|
|
|
|
|
|
|
|
const char *servicep = service.length() == 0 ? NULL : service.c_str();
|
|
|
|
|
|
|
|
*res = NULL;
|
|
|
|
int result = getaddrinfo(host.c_str(), servicep, &hints, res);
|
2013-08-26 18:59:08 +02:00
|
|
|
if (result == EAI_AGAIN) {
|
2013-06-08 18:59:11 -07:00
|
|
|
// Temporary failure. Since this already blocks, let's just try once more.
|
2014-07-13 08:15:14 +10:00
|
|
|
sleep_ms(1);
|
2013-06-08 18:59:11 -07:00
|
|
|
result = getaddrinfo(host.c_str(), servicep, &hints, res);
|
|
|
|
}
|
|
|
|
|
2013-08-26 18:59:08 +02:00
|
|
|
if (result != 0) {
|
2013-08-26 19:17:54 +02:00
|
|
|
#ifdef _WIN32
|
2013-08-26 18:59:08 +02:00
|
|
|
error = gai_strerrorA(result);
|
2013-08-26 19:17:54 +02:00
|
|
|
#else
|
|
|
|
error = gai_strerror(result);
|
|
|
|
#endif
|
2013-06-08 18:59:11 -07:00
|
|
|
if (*res != NULL)
|
|
|
|
freeaddrinfo(*res);
|
|
|
|
*res = NULL;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void DNSResolveFree(addrinfo *res)
|
|
|
|
{
|
2013-06-24 19:37:47 -07:00
|
|
|
if (res != NULL)
|
|
|
|
freeaddrinfo(res);
|
2013-06-08 18:59:11 -07:00
|
|
|
}
|
|
|
|
|
2013-03-04 14:48:18 -08:00
|
|
|
int inet_pton(int af, const char* src, void* dst)
|
|
|
|
{
|
|
|
|
if (af == AF_INET)
|
|
|
|
{
|
2013-03-04 16:10:46 -08:00
|
|
|
unsigned char *ip = (unsigned char *)dst;
|
2013-03-04 14:48:18 -08:00
|
|
|
int k = 0, x = 0;
|
|
|
|
char ch;
|
|
|
|
for (int i = 0; (ch = src[i]) != 0; i++)
|
|
|
|
{
|
|
|
|
if (ch == '.')
|
|
|
|
{
|
|
|
|
ip[k++] = x;
|
|
|
|
if (k == 4)
|
|
|
|
return 0;
|
|
|
|
x = 0;
|
|
|
|
}
|
|
|
|
else if (ch < '0' || ch > '9')
|
|
|
|
return 0;
|
|
|
|
else
|
|
|
|
x = x * 10 + ch - '0';
|
|
|
|
if (x > 255)
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
ip[k++] = x;
|
|
|
|
if (k != 4)
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
else if (af == AF_INET6)
|
|
|
|
{
|
|
|
|
unsigned short* ip = ( unsigned short* )dst;
|
|
|
|
int i;
|
|
|
|
for (i = 0; i < 8; i++) ip[i] = 0;
|
|
|
|
int k = 0;
|
|
|
|
unsigned int x = 0;
|
|
|
|
char ch;
|
|
|
|
int marknum = 0;
|
|
|
|
for (i = 0; src[i] != 0; i++)
|
|
|
|
{
|
|
|
|
if (src[i] == ':')
|
|
|
|
marknum++;
|
|
|
|
}
|
|
|
|
for (i = 0; (ch = src[i]) != 0; i++)
|
|
|
|
{
|
|
|
|
if (ch == ':')
|
|
|
|
{
|
|
|
|
x = ((x & 0xFF00) >> 8) | ((x & 0x00FF) << 8);
|
|
|
|
ip[k++] = x;
|
|
|
|
if (k == 8)
|
|
|
|
return 0;
|
|
|
|
x = 0;
|
|
|
|
if (i > 0 && src[i - 1] == ':')
|
|
|
|
k += 7 - marknum;
|
|
|
|
}
|
|
|
|
else if (ch >= '0' && ch <= '9')
|
|
|
|
x = x * 16 + ch - '0';
|
|
|
|
else if (ch >= 'a' && ch <= 'f')
|
|
|
|
x = x * 16 + ch - 'a' + 10;
|
|
|
|
else if (ch >= 'A' && ch <= 'F')
|
|
|
|
x = x * 16 + ch - 'A' + 10;
|
|
|
|
else
|
|
|
|
return 0;
|
|
|
|
if (x > 0xFFFF)
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
x = ((x & 0xFF00) >> 8) | ((x & 0x00FF) << 8);
|
|
|
|
ip[k++] = x;
|
|
|
|
if (k != 8)
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2012-06-03 19:01:08 +02:00
|
|
|
}
|