2013-06-26 15:02:42 -04:00
|
|
|
/* network.h
|
|
|
|
*
|
|
|
|
* Functions for the core networking.
|
|
|
|
*
|
|
|
|
|
|
|
|
Copyright (C) 2013 Tox project All Rights Reserved.
|
2013-06-26 09:56:15 -04:00
|
|
|
|
2013-06-26 15:02:42 -04:00
|
|
|
This file is part of Tox.
|
|
|
|
|
|
|
|
Tox is free software: you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
Tox 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 General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with Tox. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
*/
|
2013-06-26 09:56:15 -04:00
|
|
|
|
|
|
|
#include "network.h"
|
|
|
|
|
2013-06-27 07:37:06 -04:00
|
|
|
|
2013-07-03 16:35:44 -04:00
|
|
|
//returns current UNIX time in microseconds (us).
|
2013-06-27 07:37:06 -04:00
|
|
|
uint64_t current_time()
|
|
|
|
{
|
|
|
|
uint64_t time;
|
|
|
|
#ifdef WIN32
|
2013-07-03 16:35:44 -04:00
|
|
|
//This probably works fine
|
|
|
|
FILETIME ft;
|
|
|
|
GetSystemTimeAsFileTime(&ft);
|
|
|
|
time = ft.dwHighDateTime;
|
|
|
|
time <<=32;
|
|
|
|
time |= ft.dwLowDateTime;
|
|
|
|
time -= 116444736000000000UL;
|
|
|
|
return time/10;
|
2013-06-27 07:37:06 -04:00
|
|
|
#else
|
|
|
|
struct timeval a;
|
|
|
|
gettimeofday(&a, NULL);
|
|
|
|
time = 1000000UL*a.tv_sec + a.tv_usec;
|
|
|
|
return time;
|
2013-07-03 16:35:44 -04:00
|
|
|
#endif
|
|
|
|
|
2013-06-27 07:37:06 -04:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2013-07-03 09:45:01 -04:00
|
|
|
uint32_t random_int()
|
2013-06-27 17:19:09 -04:00
|
|
|
{
|
|
|
|
#ifdef WIN32
|
2013-07-03 16:35:44 -04:00
|
|
|
//NOTE: this function comes from libsodium
|
|
|
|
return randombytes_random();
|
2013-06-27 17:19:09 -04:00
|
|
|
#else
|
|
|
|
return random();
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2013-06-26 09:56:15 -04:00
|
|
|
//our UDP socket, a global variable.
|
|
|
|
static int sock;
|
|
|
|
|
|
|
|
//Basic network functions:
|
|
|
|
//Function to send packet(data) of length length to ip_port
|
2013-07-05 17:00:39 -04:00
|
|
|
int sendpacket(IP_Port ip_port, uint8_t * data, uint32_t length)
|
2013-06-26 09:56:15 -04:00
|
|
|
{
|
|
|
|
ADDR addr = {AF_INET, ip_port.port, ip_port.ip};
|
2013-07-05 17:00:39 -04:00
|
|
|
return sendto(sock,(char *) data, length, 0, (struct sockaddr *)&addr, sizeof(addr));
|
2013-06-26 09:56:15 -04:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2013-06-28 09:15:19 -04:00
|
|
|
//Function to receive data, ip and port of sender is put into ip_port
|
2013-06-26 09:56:15 -04:00
|
|
|
//the packet data into data
|
|
|
|
//the packet length into length.
|
|
|
|
//dump all empty packets.
|
2013-07-05 17:00:39 -04:00
|
|
|
int recievepacket(IP_Port * ip_port, uint8_t * data, uint32_t * length)
|
2013-06-26 09:56:15 -04:00
|
|
|
{
|
|
|
|
ADDR addr;
|
2013-07-03 16:35:44 -04:00
|
|
|
#ifdef WIN32
|
|
|
|
int addrlen = sizeof(addr);
|
|
|
|
#else
|
2013-06-26 09:56:15 -04:00
|
|
|
uint32_t addrlen = sizeof(addr);
|
2013-07-03 16:35:44 -04:00
|
|
|
#endif
|
2013-07-05 17:00:39 -04:00
|
|
|
(*(int32_t *)length) = recvfrom(sock,(char *) data, MAX_UDP_PACKET_SIZE, 0, (struct sockaddr *)&addr, &addrlen);
|
2013-07-01 08:54:21 -04:00
|
|
|
if(*(int32_t *)length <= 0)
|
2013-06-26 09:56:15 -04:00
|
|
|
{
|
2013-06-28 09:15:19 -04:00
|
|
|
//nothing received
|
2013-06-26 09:56:15 -04:00
|
|
|
//or empty packet
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
ip_port->ip = addr.ip;
|
|
|
|
ip_port->port = addr.port;
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
//initialize networking
|
|
|
|
//bind to ip and port
|
|
|
|
//ip must be in network order EX: 127.0.0.1 = (7F000001)
|
|
|
|
//port is in host byte order (this means don't worry about it)
|
|
|
|
//returns 0 if no problems
|
|
|
|
//TODO: add something to check if there are errors
|
|
|
|
int init_networking(IP ip ,uint16_t port)
|
|
|
|
{
|
|
|
|
#ifdef WIN32
|
|
|
|
WSADATA wsaData;
|
|
|
|
if(WSAStartup(MAKEWORD(2,2), &wsaData) != NO_ERROR)
|
|
|
|
{
|
|
|
|
return -1;
|
|
|
|
}
|
2013-06-27 17:19:09 -04:00
|
|
|
|
|
|
|
#else
|
|
|
|
srandom((uint32_t)current_time());
|
2013-06-26 09:56:15 -04:00
|
|
|
#endif
|
2013-06-27 17:19:09 -04:00
|
|
|
srand((uint32_t)current_time());
|
2013-06-26 09:56:15 -04:00
|
|
|
|
|
|
|
//initialize our socket
|
|
|
|
sock = socket(AF_INET, SOCK_DGRAM, IPPROTO_UDP);
|
|
|
|
|
2013-06-28 09:15:19 -04:00
|
|
|
//Functions to increase the size of the send and receive UDP buffers
|
2013-06-26 15:05:15 -04:00
|
|
|
//NOTE: uncomment if necessary
|
2013-06-26 14:07:01 -04:00
|
|
|
/*
|
|
|
|
int n = 1024 * 1024 * 2;
|
|
|
|
if(setsockopt(sock, SOL_SOCKET, SO_RCVBUF, (char*)&n, sizeof(n)) == -1)
|
|
|
|
{
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(setsockopt(sock, SOL_SOCKET, SO_SNDBUF, (char*)&n, sizeof(n)) == -1)
|
|
|
|
{
|
|
|
|
return -1;
|
|
|
|
}*/
|
|
|
|
|
2013-06-26 09:56:15 -04:00
|
|
|
//Set socket nonblocking
|
|
|
|
#ifdef WIN32
|
|
|
|
//I think this works for windows
|
|
|
|
u_long mode = 1;
|
|
|
|
//ioctl(sock, FIONBIO, &mode);
|
|
|
|
ioctlsocket(sock, FIONBIO, &mode);
|
|
|
|
#else
|
|
|
|
fcntl(sock, F_SETFL, O_NONBLOCK, 1);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
//Bind our socket to port PORT and address 0.0.0.0
|
|
|
|
ADDR addr = {AF_INET, htons(port), ip};
|
|
|
|
bind(sock, (struct sockaddr*)&addr, sizeof(addr));
|
|
|
|
return 0;
|
|
|
|
|
2013-06-27 10:01:31 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
//function to cleanup networking stuff
|
|
|
|
void shutdown_networking()
|
|
|
|
{
|
|
|
|
#ifdef WIN32
|
|
|
|
WSACleanup();
|
|
|
|
#endif
|
|
|
|
return;
|
2013-06-26 09:56:15 -04:00
|
|
|
}
|