You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
tdegames/kenolaba/Network.cpp

194 lines
4.4 KiB

#include <config.h>
#include "Network.h"
#include <unistd.h>
#include <stdio.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <netdb.h>
Listener::Listener(const char* h, int p, struct sockaddr_in s,bool r)
{
if (h==0)
host[0]=0;
else {
int l = strlen(h);
if (l>99) l=99;
strncpy(host, h, l);
host[l] = 0;
}
port = p;
sin = s;
reachable = r;
}
Network::Network(int port)
{
struct sockaddr_in name;
int i,j;
listeners.setAutoDelete(TRUE);
fd = ::socket (PF_INET, SOCK_STREAM, 0);
if (fd<0) return;
for(i = 0; i<5;i++) {
name.sin_family = AF_INET;
name.sin_port = htons (port+i);
name.sin_addr.s_addr = htonl (INADDR_ANY);
if (bind (fd, (struct sockaddr *) &name, sizeof (name)) >= 0)
break;
// printf("...Port %d in use\n", port+i);
}
mySin = name;
// printf("I'm using Port %d\n", port+i);
if (i==5) {
printf("Error in bind to port %d\n", port);
close(fd);
fd = -1;
return;
}
for(j = 0; j<i;j++)
addListener("127.0.0.1", port+j);
if (::listen(fd,5)<0) {
printf("Error in listen\n");
close(fd);
fd = -1;
return;
}
sn = new QSocketNotifier( fd, QSocketNotifier::Read );
QObject::connect( sn, SIGNAL(activated(int)),
this, SLOT(gotConnection()) );
}
Network::~Network()
{
if (fd<0) return;
close(fd);
char tmp[50];
int len = sprintf(tmp, "unreg %d", ntohs(mySin.sin_port));
Listener* l;
for(l=listeners.first(); l!=0; l=listeners.next()) {
if (l->reachable)
sendString( l->sin, tmp, len);
}
listeners.clear();
delete sn;
}
void Network::gotConnection()
{
static char tmp[1024];
int len=0;
struct sockaddr_in sin;
kde_socklen_t sz = sizeof (sin);
// printf("GotConnection: ");
int s = accept(fd,(struct sockaddr *)&sin, &sz);
if (s<0) {
printf("Error in accept\n");
return;
}
while(read(s, tmp+len, 1)==1) len++;
close(s);
tmp[len]=0; len++;
// printf("Got: '%s'\n",tmp);
if (strncmp(tmp,"reg ",4)==0) {
int port = atoi(tmp+4);
sin.sin_port = htons( port );
Listener *l = new Listener(0,0,sin);
// printf("Reg of 0x%x:%d\n",
// ntohl(sin.sin_addr.s_addr ), ntohs(sin.sin_port));
listeners.append(l);
return;
}
if (strncmp(tmp,"unreg ",6)==0) {
int port = atoi(tmp+6);
sin.sin_port = htons( port );
Listener* l;
for(l=listeners.first(); l!=0; l=listeners.next())
if (l->sin.sin_addr.s_addr == sin.sin_addr.s_addr &&
l->sin.sin_port == sin.sin_port) break;
if (l==0) {
printf("Error: UnReg of 0x%x:%d. Not Found\n",
ntohl(sin.sin_addr.s_addr), ntohs(sin.sin_port));
return;
}
listeners.remove(l);
// printf("UnReg of 0x%x:%d\n",
// ntohl(sin.sin_addr.s_addr), ntohs(sin.sin_port));
return;
}
if (strncmp(tmp,"pos ",4)==0) {
emit gotPosition(tmp+4);
}
}
void Network::addListener(const char* host, int port)
{
struct hostent *hostinfo;
struct sockaddr_in name;
memset(&name, 0, sizeof(struct sockaddr_in));
name.sin_family = AF_INET;
name.sin_port = htons (port);
hostinfo = gethostbyname (host);
if (hostinfo == NULL) {
printf ("Error in addListener: Unknown host %s.\n", host);
return;
}
name.sin_addr = *(struct in_addr *) hostinfo->h_addr;
Listener *l = new Listener(host,port,name);
// printf("Added Listener %s, 0x%x:%d\n",
// host, ntohl(name.sin_addr.s_addr), ntohs(name.sin_port));
listeners.append(l);
char tmp[50];
int len = sprintf(tmp, "reg %d", ntohs(mySin.sin_port));
if (!sendString( name, tmp, len))
listeners.remove(l);
}
void Network::broadcast(const char* pos)
{
char tmp[1024];
int len = sprintf(tmp,"pos %s", pos);
for(Listener* l=listeners.first(); l!=0; l=listeners.next())
if (l->reachable)
l->reachable = sendString(l->sin, tmp, len);
}
bool Network::sendString(struct sockaddr_in sin, char* str, int len)
{
int s = ::socket (PF_INET, SOCK_STREAM, 0);
if (s<0) {
printf("Error in sendString/socket ??\n");
return false;
}
if (::connect (s, (struct sockaddr *)&sin, sizeof (sin)) <0) {
printf("Error in sendString/connect to socket 0x%x:%d\n",
ntohl(sin.sin_addr.s_addr), ntohs(sin.sin_port) );
return false;
}
write(s, str, len);
close(s);
// printf("Send '%s' to 0x%x:%d\n", str,
// ntohl(sin.sin_addr.s_addr), ntohs(sin.sin_port) );
return true;
}
#include "Network.moc"