2009-01-19 07:44:19 +01:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <sys/socket.h>
|
|
|
|
#include <netinet/in.h>
|
2009-01-24 16:48:43 +01:00
|
|
|
#include <netdb.h>
|
2009-01-19 07:44:19 +01:00
|
|
|
|
2009-01-29 19:04:31 +01:00
|
|
|
static int set_sock_opts(int sock)
|
|
|
|
{
|
|
|
|
struct timeval tv;
|
|
|
|
int d = 1;
|
|
|
|
|
|
|
|
memset(&tv, 0, sizeof(tv));
|
|
|
|
tv.tv_sec = 2;
|
|
|
|
setsockopt(sock, SOL_SOCKET, SO_RCVTIMEO,
|
|
|
|
&tv, sizeof(struct timeval));
|
|
|
|
setsockopt(sock, SOL_SOCKET, SO_SNDTIMEO,
|
|
|
|
&tv, sizeof(struct timeval));
|
|
|
|
setsockopt(sock,SOL_SOCKET,SO_REUSEADDR, &d, sizeof(int));
|
|
|
|
}
|
|
|
|
|
2009-01-19 07:44:19 +01:00
|
|
|
/* From glibc docs */
|
|
|
|
static int make_socket (uint16_t port)
|
|
|
|
{
|
|
|
|
struct sockaddr_in name;
|
2009-01-24 21:57:23 +01:00
|
|
|
int sock;
|
2009-01-19 07:44:19 +01:00
|
|
|
|
|
|
|
/* Create the socket. */
|
|
|
|
sock = socket (PF_INET, SOCK_STREAM, 0);
|
|
|
|
if (sock < 0)
|
|
|
|
{
|
|
|
|
perror ("socket");
|
|
|
|
exit (EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
|
2009-01-29 19:04:31 +01:00
|
|
|
set_sock_opts(sock);
|
2009-01-24 21:57:23 +01:00
|
|
|
|
2009-01-19 07:44:19 +01:00
|
|
|
/* Give the socket a name. */
|
|
|
|
name.sin_family = AF_INET;
|
|
|
|
name.sin_port = htons (port);
|
|
|
|
name.sin_addr.s_addr = htonl (INADDR_ANY);
|
|
|
|
if (bind (sock, (struct sockaddr *) &name, sizeof (name)) < 0)
|
|
|
|
{
|
|
|
|
perror ("bind");
|
|
|
|
exit (1);
|
|
|
|
}
|
|
|
|
|
|
|
|
return sock;
|
|
|
|
}
|
|
|
|
|
2009-01-24 16:48:43 +01:00
|
|
|
bool init_sockaddr (struct sockaddr_in *name,
|
|
|
|
const char *hostname, uint16_t port)
|
|
|
|
{
|
|
|
|
struct hostent *hostinfo;
|
|
|
|
|
|
|
|
name->sin_family = AF_INET;
|
|
|
|
name->sin_port = htons (port);
|
|
|
|
hostinfo = gethostbyname (hostname);
|
|
|
|
if (hostinfo == NULL)
|
|
|
|
{
|
|
|
|
fprintf (stderr, "Unknown host %s.\n", hostname);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
name->sin_addr = *(struct in_addr *) hostinfo->h_addr;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
2009-01-19 07:44:19 +01:00
|
|
|
|
2009-01-24 16:48:43 +01:00
|
|
|
|
2009-01-25 11:07:51 +01:00
|
|
|
NetworkServer::NetworkServer(int port) : Network()
|
2009-01-19 07:44:19 +01:00
|
|
|
{
|
|
|
|
this->n_clients = 0;
|
2009-01-24 16:48:43 +01:00
|
|
|
this->listen_sock = make_socket(port);
|
2009-01-19 07:44:19 +01:00
|
|
|
|
2009-01-24 16:48:43 +01:00
|
|
|
if (listen(this->listen_sock, MAX_NETWORK_CLIENTS) < 0)
|
2009-01-19 07:44:19 +01:00
|
|
|
{
|
|
|
|
perror("listen");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-01-24 16:48:43 +01:00
|
|
|
bool NetworkServer::CheckNewConnection()
|
2009-01-19 07:44:19 +01:00
|
|
|
{
|
|
|
|
struct timeval tv;
|
|
|
|
struct sockaddr_in client_name;
|
|
|
|
size_t size;
|
|
|
|
int client_sock;
|
2009-01-24 16:48:43 +01:00
|
|
|
fd_set listen_fds;
|
|
|
|
|
|
|
|
/* No more than that thanks... */
|
|
|
|
if (this->n_clients >= MAX_NETWORK_CLIENTS)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
FD_ZERO(&listen_fds);
|
|
|
|
FD_SET(this->listen_sock, &listen_fds);
|
2009-01-19 07:44:19 +01:00
|
|
|
|
|
|
|
/* If something connects, create a new client */
|
|
|
|
memset(&tv, 0, sizeof(tv));
|
2009-01-24 16:48:43 +01:00
|
|
|
int v = select(this->listen_sock + 1, &listen_fds, NULL, NULL, &tv);
|
|
|
|
|
|
|
|
if ( v < 0)
|
|
|
|
{
|
|
|
|
perror("select");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
else if ( v == 0 )
|
|
|
|
return false;
|
2009-01-19 07:44:19 +01:00
|
|
|
|
2009-01-24 21:57:23 +01:00
|
|
|
size = sizeof(client_name);
|
2009-01-19 07:44:19 +01:00
|
|
|
client_sock = accept(this->listen_sock, (struct sockaddr*)&client_name, &size);
|
|
|
|
if (client_sock < 0)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "Accepting client failed\n");
|
2009-01-24 16:48:43 +01:00
|
|
|
return false;
|
2009-01-19 07:44:19 +01:00
|
|
|
}
|
|
|
|
|
2009-01-24 16:48:43 +01:00
|
|
|
/* And add the new one! */
|
|
|
|
this->AddClient(client_sock);
|
2009-01-19 07:44:19 +01:00
|
|
|
|
2009-01-24 16:48:43 +01:00
|
|
|
return true;
|
|
|
|
}
|
2009-01-19 07:44:19 +01:00
|
|
|
|
2009-01-24 16:48:43 +01:00
|
|
|
NetworkClient::NetworkClient(const char *hostname, int port)
|
2009-01-19 07:44:19 +01:00
|
|
|
{
|
2009-01-24 16:48:43 +01:00
|
|
|
/* Again from glibc docs */
|
|
|
|
struct sockaddr_in servername;
|
|
|
|
|
2009-01-30 16:37:15 +01:00
|
|
|
this->Init();
|
|
|
|
|
2009-01-24 16:48:43 +01:00
|
|
|
/* Create the socket. */
|
|
|
|
this->sock = socket (PF_INET, SOCK_STREAM, 0);
|
|
|
|
if (this->sock < 0)
|
|
|
|
{
|
|
|
|
perror ("socket (client)");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2009-01-29 19:04:31 +01:00
|
|
|
set_sock_opts(sock);
|
|
|
|
|
2009-01-24 16:48:43 +01:00
|
|
|
/* Connect to the server. */
|
|
|
|
init_sockaddr (&servername, hostname, port);
|
|
|
|
if (connect(sock, (struct sockaddr *) &servername,
|
|
|
|
sizeof (servername)) != 0)
|
|
|
|
{
|
|
|
|
perror ("connect (client)");
|
|
|
|
return;
|
|
|
|
}
|
2009-01-19 07:44:19 +01:00
|
|
|
}
|
|
|
|
|
2009-01-26 22:00:23 +01:00
|
|
|
bool Network::ReceiveData(void *dst, int sock, size_t sz)
|
2009-01-19 07:44:19 +01:00
|
|
|
{
|
2009-01-26 22:00:23 +01:00
|
|
|
size_t received_sz = 0;
|
2009-01-19 07:44:19 +01:00
|
|
|
|
2009-01-26 22:00:23 +01:00
|
|
|
while (received_sz < sz)
|
2009-01-19 07:44:19 +01:00
|
|
|
{
|
2009-01-26 22:00:23 +01:00
|
|
|
int v = read(sock, dst, sz);
|
|
|
|
|
|
|
|
if (v < 0)
|
|
|
|
return false;
|
|
|
|
received_sz += v;
|
2009-01-19 07:44:19 +01:00
|
|
|
}
|
2009-01-30 18:49:47 +01:00
|
|
|
this->traffic += received_sz;
|
2009-01-19 07:44:19 +01:00
|
|
|
|
2009-01-26 22:00:23 +01:00
|
|
|
return sz > 0;
|
|
|
|
}
|
2009-01-19 07:44:19 +01:00
|
|
|
|
2009-01-26 22:00:23 +01:00
|
|
|
bool Network::SendData(void *src, int sock, size_t sz)
|
|
|
|
{
|
|
|
|
size_t sent_sz = 0;
|
|
|
|
|
|
|
|
while (sent_sz < sz)
|
|
|
|
{
|
|
|
|
int v = write(sock, (void*)src, sz);
|
|
|
|
|
|
|
|
if (v < 0)
|
|
|
|
return false;
|
|
|
|
sent_sz += v;
|
|
|
|
}
|
2009-01-19 07:44:19 +01:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2009-01-26 22:00:23 +01:00
|
|
|
bool Network::Select(int sock, struct timeval *tv)
|
2009-01-19 07:44:19 +01:00
|
|
|
{
|
2009-01-26 22:00:23 +01:00
|
|
|
fd_set fds;
|
|
|
|
int v;
|
2009-01-19 07:44:19 +01:00
|
|
|
|
2009-01-26 22:00:23 +01:00
|
|
|
FD_ZERO(&fds);
|
|
|
|
FD_SET(sock, &fds);
|
2009-01-24 21:57:23 +01:00
|
|
|
|
2009-01-26 22:00:23 +01:00
|
|
|
v = select(sock + 1, &fds, NULL, NULL, tv);
|
|
|
|
if (v < 0)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "Select failed\n");
|
|
|
|
return false;
|
|
|
|
}
|
2009-01-24 16:48:43 +01:00
|
|
|
|
2009-01-26 22:00:23 +01:00
|
|
|
/* v is 0 if the sock is not ready */
|
|
|
|
return v > 0;
|
2009-01-19 07:44:19 +01:00
|
|
|
}
|
2009-01-29 21:35:06 +01:00
|
|
|
|
|
|
|
void Network::CloseSocket(int sock)
|
|
|
|
{
|
|
|
|
close(sock);
|
|
|
|
}
|