2008-08-06 18:36:30 +00:00
|
|
|
// This file is part of BOINC.
|
2005-03-07 06:09:04 +00:00
|
|
|
// http://boinc.berkeley.edu
|
2008-08-06 18:36:30 +00:00
|
|
|
// Copyright (C) 2008 University of California
|
2005-03-07 06:09:04 +00:00
|
|
|
//
|
2008-08-06 18:36:30 +00:00
|
|
|
// BOINC is free software; you can redistribute it and/or modify it
|
|
|
|
// under the terms of the GNU Lesser General Public License
|
|
|
|
// as published by the Free Software Foundation,
|
|
|
|
// either version 3 of the License, or (at your option) any later version.
|
2005-03-07 06:09:04 +00:00
|
|
|
//
|
2008-08-06 18:36:30 +00:00
|
|
|
// BOINC is distributed in the hope that it will be useful,
|
2005-03-07 06:09:04 +00:00
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
|
|
|
// See the GNU Lesser General Public License for more details.
|
|
|
|
//
|
2008-08-06 18:36:30 +00:00
|
|
|
// You should have received a copy of the GNU Lesser General Public License
|
|
|
|
// along with BOINC. If not, see <http://www.gnu.org/licenses/>.
|
2005-03-07 06:09:04 +00:00
|
|
|
|
2020-06-04 07:48:21 +00:00
|
|
|
#if defined(_WIN32)
|
2005-07-14 16:46:38 +00:00
|
|
|
#include "boinc_win.h"
|
2010-09-15 20:33:44 +00:00
|
|
|
#include <fcntl.h>
|
2010-05-11 19:10:29 +00:00
|
|
|
#else
|
2005-11-21 18:34:44 +00:00
|
|
|
#include "config.h"
|
2011-09-27 19:45:27 +00:00
|
|
|
#if HAVE_UNISTD_H
|
2005-03-07 21:19:09 +00:00
|
|
|
#include <unistd.h>
|
2009-02-26 00:23:23 +00:00
|
|
|
#endif
|
2005-04-08 23:40:50 +00:00
|
|
|
#include <cstdio>
|
2009-02-26 00:23:23 +00:00
|
|
|
#include <cstdlib>
|
2005-03-07 21:19:09 +00:00
|
|
|
#include <sys/socket.h>
|
2005-04-10 00:16:20 +00:00
|
|
|
#include <netinet/in.h>
|
|
|
|
#include <netinet/tcp.h>
|
|
|
|
#include <arpa/inet.h>
|
2012-11-19 06:50:11 +00:00
|
|
|
#include <resolv.h>
|
2005-04-10 00:16:20 +00:00
|
|
|
#include <netdb.h>
|
2005-04-08 23:40:50 +00:00
|
|
|
#include <fcntl.h>
|
2005-12-16 03:35:15 +00:00
|
|
|
#include <errno.h>
|
2005-03-07 21:19:09 +00:00
|
|
|
#endif
|
|
|
|
|
2012-08-01 20:04:05 +00:00
|
|
|
#include "error_numbers.h"
|
2014-11-14 21:07:24 +00:00
|
|
|
#include "str_util.h"
|
|
|
|
#include "util.h"
|
2012-08-01 20:04:05 +00:00
|
|
|
|
2012-08-11 05:47:18 +00:00
|
|
|
#include "network.h"
|
|
|
|
|
2009-02-26 00:23:23 +00:00
|
|
|
using std::perror;
|
|
|
|
using std::sprintf;
|
|
|
|
|
2005-06-07 21:46:11 +00:00
|
|
|
const char* socket_error_str() {
|
2005-05-14 19:45:24 +00:00
|
|
|
static char buf[80];
|
2005-12-16 03:35:15 +00:00
|
|
|
#if defined(_WIN32) && defined(USE_WINSOCK)
|
2005-05-14 19:45:24 +00:00
|
|
|
int e = WSAGetLastError();
|
|
|
|
switch (e) {
|
|
|
|
case WSANOTINITIALISED:
|
|
|
|
return "WSA not initialized";
|
|
|
|
case WSAENETDOWN:
|
|
|
|
return "the network subsystem has failed";
|
|
|
|
case WSAHOST_NOT_FOUND:
|
|
|
|
return "host name not found";
|
|
|
|
case WSATRY_AGAIN:
|
|
|
|
return "no response from server";
|
|
|
|
case WSANO_RECOVERY:
|
|
|
|
return "a nonrecoverable error occurred";
|
|
|
|
case WSANO_DATA:
|
|
|
|
return "valid name, no data record of requested type";
|
|
|
|
case WSAEINPROGRESS:
|
|
|
|
return "a blocking socket call in progress";
|
|
|
|
case WSAEFAULT:
|
|
|
|
return "invalid part of user address space";
|
|
|
|
case WSAEINTR:
|
|
|
|
return "a blocking socket call was canceled";
|
|
|
|
case WSAENOTSOCK:
|
|
|
|
return "not a socket";
|
|
|
|
}
|
2016-02-17 23:50:28 +00:00
|
|
|
snprintf(buf, sizeof(buf), "error %d", e);
|
2005-05-14 19:45:24 +00:00
|
|
|
return buf;
|
|
|
|
#else
|
|
|
|
switch (h_errno) {
|
|
|
|
case HOST_NOT_FOUND:
|
|
|
|
return "host not found";
|
|
|
|
case NO_DATA:
|
|
|
|
return "valid name, no data record of requested type";
|
|
|
|
case NO_RECOVERY:
|
|
|
|
return "a nonrecoverable error occurred";
|
|
|
|
case TRY_AGAIN:
|
|
|
|
return "host not found or server failure";
|
2005-12-16 03:35:15 +00:00
|
|
|
#ifdef NETDB_INTERNAL
|
|
|
|
case NETDB_INTERNAL:
|
2016-02-17 23:50:28 +00:00
|
|
|
snprintf(buf, sizeof(buf), "network internal error %d", errno);
|
2010-05-31 21:51:09 +00:00
|
|
|
return buf;
|
2005-12-16 03:35:15 +00:00
|
|
|
#endif
|
2005-05-14 19:45:24 +00:00
|
|
|
}
|
2016-02-17 23:50:28 +00:00
|
|
|
snprintf(buf, sizeof(buf), "error %d", h_errno);
|
2005-05-14 19:45:24 +00:00
|
|
|
return buf;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2010-09-15 17:41:25 +00:00
|
|
|
bool is_localhost(sockaddr_storage& s) {
|
2010-09-15 20:33:44 +00:00
|
|
|
#ifdef _WIN32
|
|
|
|
if (ntohl(s.sin_addr.s_addr) == 0x7f000001) return true;
|
|
|
|
#else
|
2010-09-15 17:41:25 +00:00
|
|
|
switch (s.ss_family) {
|
2013-03-04 13:05:41 +00:00
|
|
|
case AF_INET: {
|
2010-09-15 17:41:25 +00:00
|
|
|
sockaddr_in* sin = (sockaddr_in*)&s;
|
|
|
|
return (ntohl(sin->sin_addr.s_addr) == 0x7f000001);
|
2013-03-04 13:05:41 +00:00
|
|
|
break;
|
2010-09-15 17:41:25 +00:00
|
|
|
}
|
2013-03-04 13:05:41 +00:00
|
|
|
case AF_INET6: {
|
2010-09-15 17:41:25 +00:00
|
|
|
sockaddr_in6* sin = (sockaddr_in6*)&s;
|
|
|
|
char buf[256];
|
2013-05-22 20:56:48 +00:00
|
|
|
inet_ntop(AF_INET6, (void*)(&sin->sin6_addr), buf, sizeof(buf));
|
2010-09-15 17:41:25 +00:00
|
|
|
return (strcmp(buf, "::1") == 0);
|
2013-03-04 13:05:41 +00:00
|
|
|
break;
|
2010-09-15 17:41:25 +00:00
|
|
|
}
|
|
|
|
}
|
2010-09-15 20:33:44 +00:00
|
|
|
#endif
|
2010-09-15 17:41:25 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool same_ip_addr(sockaddr_storage& s1, sockaddr_storage& s2) {
|
2010-09-15 20:33:44 +00:00
|
|
|
#ifdef _WIN32
|
|
|
|
return (s1.sin_addr.s_addr == s2.sin_addr.s_addr);
|
|
|
|
#else
|
2010-09-15 17:41:25 +00:00
|
|
|
if (s1.ss_family != s2.ss_family) return false;
|
|
|
|
switch (s1.ss_family) {
|
2013-03-04 13:05:41 +00:00
|
|
|
case AF_INET: {
|
2010-09-15 17:41:25 +00:00
|
|
|
sockaddr_in* sin1 = (sockaddr_in*)&s1;
|
|
|
|
sockaddr_in* sin2 = (sockaddr_in*)&s2;
|
|
|
|
return (memcmp((void*)(&sin1->sin_addr), (void*)(&sin2->sin_addr), sizeof(in_addr)) == 0);
|
|
|
|
break;
|
|
|
|
}
|
2013-03-04 13:05:41 +00:00
|
|
|
case AF_INET6: {
|
2010-09-15 17:41:25 +00:00
|
|
|
sockaddr_in6* sin1 = (sockaddr_in6*)&s1;
|
|
|
|
sockaddr_in6* sin2 = (sockaddr_in6*)&s2;
|
|
|
|
return (memcmp((void*)(&sin1->sin6_addr), (void*)(&sin2->sin6_addr), sizeof(in6_addr)) == 0);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
2010-09-15 20:33:44 +00:00
|
|
|
#endif
|
2010-09-15 17:41:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int resolve_hostname(const char* hostname, sockaddr_storage &ip_addr) {
|
2010-09-15 20:33:44 +00:00
|
|
|
#ifdef _WIN32
|
|
|
|
hostent* hep;
|
|
|
|
hep = gethostbyname(hostname);
|
|
|
|
if (!hep) {
|
|
|
|
return ERR_GETHOSTBYNAME;
|
|
|
|
}
|
2014-08-19 07:36:44 +00:00
|
|
|
for (int i=0; ; i++) {
|
|
|
|
if (!hep->h_addr_list[i]) break;
|
|
|
|
ip_addr.sin_family = AF_INET;
|
|
|
|
ip_addr.sin_addr.s_addr = *(int*)hep->h_addr_list[i];
|
|
|
|
if ((ip_addr.sin_addr.s_addr&0xff) != 0x7f) return 0; // look for non-loopback addr
|
|
|
|
}
|
2010-09-15 20:33:44 +00:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
#else
|
2010-09-15 17:41:25 +00:00
|
|
|
struct addrinfo *res, hints;
|
2005-04-10 00:16:20 +00:00
|
|
|
|
2010-09-15 17:41:25 +00:00
|
|
|
memset(&hints, 0, sizeof(hints));
|
2010-10-07 18:08:20 +00:00
|
|
|
hints.ai_family = AF_INET;
|
2010-09-15 17:41:25 +00:00
|
|
|
hints.ai_socktype = SOCK_STREAM;
|
|
|
|
hints.ai_protocol = IPPROTO_TCP;
|
2016-05-02 09:20:00 +00:00
|
|
|
char buf[512];
|
|
|
|
snprintf(buf, sizeof(buf), "%s getaddrinfo(%s)", time_to_string(dtime()), hostname);
|
2010-09-15 17:41:25 +00:00
|
|
|
int retval = getaddrinfo(hostname, NULL, &hints, &res);
|
|
|
|
if (retval) {
|
2016-05-02 07:09:57 +00:00
|
|
|
if (retval == EAI_SYSTEM) {
|
|
|
|
perror(buf);
|
|
|
|
} else {
|
2020-04-04 18:18:21 +00:00
|
|
|
fprintf(stderr, "%s: %s\n", buf, gai_strerror(retval));
|
2016-05-02 07:09:57 +00:00
|
|
|
}
|
2014-11-14 21:07:24 +00:00
|
|
|
return ERR_GETADDRINFO;
|
2010-09-15 17:41:25 +00:00
|
|
|
}
|
2014-08-19 07:50:45 +00:00
|
|
|
struct addrinfo* aip = res;
|
|
|
|
while (aip) {
|
|
|
|
memcpy(&ip_addr, aip->ai_addr, aip->ai_addrlen);
|
|
|
|
sockaddr_in* sin = (sockaddr_in*)&ip_addr;
|
|
|
|
if ((sin->sin_addr.s_addr&0xff) != 0x7f) break;
|
|
|
|
aip = aip->ai_next;
|
2014-08-19 07:36:44 +00:00
|
|
|
}
|
2010-09-15 17:41:25 +00:00
|
|
|
freeaddrinfo(res);
|
|
|
|
return 0;
|
2010-09-15 20:33:44 +00:00
|
|
|
#endif
|
2010-09-15 17:41:25 +00:00
|
|
|
}
|
|
|
|
|
2013-03-04 13:05:41 +00:00
|
|
|
int resolve_hostname_or_ip_addr(
|
|
|
|
const char* hostname, sockaddr_storage &ip_addr
|
|
|
|
) {
|
2010-09-15 20:33:44 +00:00
|
|
|
#ifdef _WIN32 // inet_pton() only on Vista or later!!
|
|
|
|
int x = inet_addr(hostname);
|
|
|
|
if (x != -1) {
|
|
|
|
sockaddr_in* sin = (sockaddr_in*)&ip_addr;
|
|
|
|
sin->sin_family = AF_INET;
|
|
|
|
sin->sin_addr.s_addr = x;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
#else
|
2010-09-15 17:41:25 +00:00
|
|
|
int retval;
|
|
|
|
// check for IPV4 and IPV6 notation
|
2005-04-10 00:16:20 +00:00
|
|
|
//
|
2010-09-15 17:41:25 +00:00
|
|
|
sockaddr_in* sin = (sockaddr_in*)&ip_addr;
|
|
|
|
retval = inet_pton(AF_INET, hostname, &sin->sin_addr);
|
|
|
|
if (retval > 0) {
|
|
|
|
ip_addr.ss_family = AF_INET;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
sockaddr_in6* sin6 = (sockaddr_in6*)&ip_addr;
|
|
|
|
retval = inet_pton(AF_INET6, hostname, &sin6->sin6_addr);
|
|
|
|
if (retval > 0) {
|
|
|
|
ip_addr.ss_family = AF_INET6;
|
2005-04-10 00:16:20 +00:00
|
|
|
return 0;
|
|
|
|
}
|
2010-09-15 20:33:44 +00:00
|
|
|
#endif
|
2005-04-10 00:16:20 +00:00
|
|
|
|
|
|
|
// else resolve the name
|
|
|
|
//
|
2010-09-15 17:41:25 +00:00
|
|
|
return resolve_hostname(hostname, ip_addr);
|
2005-04-10 00:16:20 +00:00
|
|
|
}
|
|
|
|
|
2014-01-31 06:57:44 +00:00
|
|
|
int boinc_socket(int& fd, int protocol) {
|
|
|
|
fd = (int)socket(protocol, SOCK_STREAM, 0);
|
2005-04-08 23:40:50 +00:00
|
|
|
if (fd < 0) {
|
2014-11-14 21:07:24 +00:00
|
|
|
char buf[256];
|
2016-05-02 09:20:00 +00:00
|
|
|
snprintf(buf, sizeof(buf), "%s socket()", time_to_string(dtime()));
|
2016-02-17 23:50:28 +00:00
|
|
|
perror(buf);
|
2005-04-08 23:40:50 +00:00
|
|
|
return ERR_SOCKET;
|
|
|
|
}
|
2005-12-18 02:00:15 +00:00
|
|
|
#ifndef _WIN32
|
2015-11-04 11:02:43 +00:00
|
|
|
if (-1 == fcntl(fd, F_SETFD, FD_CLOEXEC)) {
|
|
|
|
return ERR_FCNTL;
|
|
|
|
}
|
2005-12-18 02:00:15 +00:00
|
|
|
#endif
|
2005-04-08 23:40:50 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int boinc_socket_asynch(int fd, bool asynch) {
|
|
|
|
if (asynch) {
|
2005-12-16 03:35:15 +00:00
|
|
|
#if defined(_WIN32) && defined(USE_WINSOCK)
|
2005-04-08 23:40:50 +00:00
|
|
|
unsigned long one = 1;
|
|
|
|
ioctlsocket(fd, FIONBIO, &one);
|
|
|
|
#else
|
|
|
|
int flags;
|
|
|
|
flags = fcntl(fd, F_GETFL, 0);
|
|
|
|
if (flags < 0) {
|
|
|
|
return ERR_FCNTL;
|
|
|
|
}
|
|
|
|
if (fcntl(fd, F_SETFL, flags|O_NONBLOCK) < 0 ) {
|
|
|
|
return ERR_FCNTL;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
} else {
|
2005-12-16 03:35:15 +00:00
|
|
|
#if defined(_WIN32) && defined(USE_WINSOCK)
|
2005-04-08 23:40:50 +00:00
|
|
|
unsigned long zero = 0;
|
|
|
|
ioctlsocket(fd, FIONBIO, &zero);
|
|
|
|
#else
|
|
|
|
int flags;
|
|
|
|
flags = fcntl(fd, F_GETFL, 0);
|
|
|
|
if (flags < 0) {
|
|
|
|
return ERR_FCNTL;
|
|
|
|
}
|
|
|
|
if (fcntl(fd, F_SETFL, flags&(~O_NONBLOCK)) < 0 ) {
|
|
|
|
return ERR_FCNTL;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-03-07 06:09:04 +00:00
|
|
|
void boinc_close_socket(int sock) {
|
2005-12-16 03:35:15 +00:00
|
|
|
#if defined(_WIN32) && defined(USE_WINSOCK)
|
2005-03-07 06:09:04 +00:00
|
|
|
closesocket(sock);
|
|
|
|
#else
|
|
|
|
close(sock);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
int get_socket_error(int fd) {
|
|
|
|
int n;
|
2005-12-16 03:35:15 +00:00
|
|
|
#if defined(_WIN32) && defined(USE_WINSOCK)
|
2009-06-16 21:58:38 +00:00
|
|
|
int intsize = sizeof(int);
|
2005-03-07 06:09:04 +00:00
|
|
|
getsockopt(fd, SOL_SOCKET, SO_ERROR, (char *)&n, &intsize);
|
2005-05-05 18:59:55 +00:00
|
|
|
#elif defined(__FreeBSD__)
|
|
|
|
// workaround for FreeBSD. I don't understand this.
|
|
|
|
struct sockaddr_in sin;
|
|
|
|
socklen_t sinsz = sizeof(sin);
|
|
|
|
n = getpeername(fd, (struct sockaddr *)&sin, &sinsz);
|
2005-03-07 06:09:04 +00:00
|
|
|
#else
|
2009-06-16 21:58:38 +00:00
|
|
|
socklen_t intsize = sizeof(int);
|
2015-11-04 11:16:49 +00:00
|
|
|
if (getsockopt(fd, SOL_SOCKET, SO_ERROR, (void*)&n, (socklen_t*)&intsize)) {
|
|
|
|
return errno;
|
|
|
|
}
|
2005-03-07 06:09:04 +00:00
|
|
|
#endif
|
|
|
|
return n;
|
|
|
|
}
|
2005-03-29 23:56:50 +00:00
|
|
|
|
2005-12-16 03:35:15 +00:00
|
|
|
#if defined(_WIN32) && defined(USE_WINSOCK)
|
2005-03-29 23:56:50 +00:00
|
|
|
|
2005-06-22 10:51:34 +00:00
|
|
|
int WinsockInitialize() {
|
|
|
|
WSADATA wsdata;
|
2010-05-31 21:51:09 +00:00
|
|
|
return WSAStartup(MAKEWORD(2, 0), &wsdata);
|
2005-03-29 23:56:50 +00:00
|
|
|
}
|
|
|
|
|
2005-06-22 10:51:34 +00:00
|
|
|
int WinsockCleanup() {
|
|
|
|
return WSACleanup();
|
2005-03-29 23:56:50 +00:00
|
|
|
}
|
2005-03-30 00:01:34 +00:00
|
|
|
|
2009-03-31 16:45:17 +00:00
|
|
|
#endif
|
2005-03-30 00:01:34 +00:00
|
|
|
|
2009-04-12 15:30:45 +00:00
|
|
|
void reset_dns() {
|
2012-08-04 00:27:32 +00:00
|
|
|
#if !defined(ANDROID) && !defined(_WIN32) && !defined(__APPLE__)
|
2009-05-07 13:54:51 +00:00
|
|
|
// Windows doesn't have this, and it crashes Macs
|
2009-04-12 15:30:45 +00:00
|
|
|
res_init();
|
|
|
|
#endif
|
2014-01-31 06:57:44 +00:00
|
|
|
}
|
2014-08-03 06:07:22 +00:00
|
|
|
|
|
|
|
// Get an unused port number.
|
|
|
|
// Used by vboxwrapper.
|
|
|
|
// I'm not sure if is_remote is relevant here - a port is a port, right?
|
|
|
|
//
|
|
|
|
int boinc_get_port(bool is_remote, int& port) {
|
|
|
|
sockaddr_in addr;
|
|
|
|
BOINC_SOCKLEN_T addrsize;
|
|
|
|
int sock;
|
|
|
|
int retval;
|
|
|
|
|
|
|
|
addrsize = sizeof(sockaddr_in);
|
|
|
|
|
|
|
|
memset(&addr, 0, sizeof(sockaddr_in));
|
|
|
|
addr.sin_family = AF_INET;
|
|
|
|
addr.sin_port = htons(0);
|
|
|
|
addr.sin_addr.s_addr = htonl(is_remote?INADDR_ANY:INADDR_LOOPBACK);
|
|
|
|
|
|
|
|
retval = boinc_socket(sock);
|
|
|
|
if (retval) return retval;
|
|
|
|
|
|
|
|
retval = bind(sock, (const sockaddr*)&addr, addrsize);
|
|
|
|
if (retval < 0) {
|
|
|
|
boinc_close_socket(sock);
|
|
|
|
return ERR_BIND;
|
|
|
|
}
|
|
|
|
|
2015-11-04 10:52:11 +00:00
|
|
|
retval = getsockname(sock, (sockaddr*)&addr, &addrsize);
|
|
|
|
if (retval) {
|
|
|
|
boinc_close_socket(sock);
|
|
|
|
return errno;
|
|
|
|
}
|
2014-08-03 06:07:22 +00:00
|
|
|
port = ntohs(addr.sin_port);
|
|
|
|
|
|
|
|
boinc_close_socket(sock);
|
|
|
|
return 0;
|
|
|
|
}
|