giuliomoro@44
|
1 /*
|
giuliomoro@44
|
2 * udpServer.cpp
|
giuliomoro@44
|
3 *
|
giuliomoro@44
|
4 * Created on: 19 May 2015
|
giuliomoro@44
|
5 * Author: giulio moro
|
giuliomoro@44
|
6 */
|
giuliomoro@44
|
7 #include "../include/UdpServer.h"
|
giuliomoro@44
|
8
|
giuliomoro@44
|
9 UdpServer::UdpServer(int aPort){
|
giuliomoro@44
|
10 init(aPort);
|
giuliomoro@44
|
11 };
|
giuliomoro@44
|
12 UdpServer::UdpServer(){
|
giuliomoro@44
|
13 init(0);
|
giuliomoro@44
|
14 }
|
giuliomoro@44
|
15 UdpServer::~UdpServer(){
|
giuliomoro@54
|
16 //TODO: unbind from port. AFAIK, this involves closing the socket, therefore creating the socket should become part of bindToPort
|
giuliomoro@44
|
17 };
|
giuliomoro@44
|
18 bool UdpServer::init(int aPort){
|
giuliomoro@44
|
19 enabled=true;
|
giuliomoro@121
|
20 stZeroTimeOut.tv_sec = 0; //set timeout to 0
|
giuliomoro@121
|
21 stZeroTimeOut.tv_usec = 0;
|
giuliomoro@44
|
22 inSocket=socket(AF_INET, SOCK_DGRAM, 0);
|
giuliomoro@44
|
23 if (inSocket < 0){
|
giuliomoro@44
|
24 enabled=false;
|
giuliomoro@44
|
25 }
|
giuliomoro@44
|
26 length = sizeof(server);
|
giuliomoro@44
|
27 server.sin_family=AF_INET;
|
giuliomoro@44
|
28 server.sin_addr.s_addr=INADDR_ANY;
|
giuliomoro@44
|
29 enabled=bindToPort(aPort);
|
giuliomoro@44
|
30 wasteBufferSize=2048;
|
giuliomoro@44
|
31 wasteBuffer=malloc(wasteBufferSize);
|
giuliomoro@44
|
32 return enabled;
|
giuliomoro@44
|
33 }
|
giuliomoro@44
|
34
|
giuliomoro@44
|
35 bool UdpServer::bindToPort(int aPort){
|
giuliomoro@44
|
36 port=aPort;
|
giuliomoro@53
|
37 if(port<1){
|
giuliomoro@53
|
38 enabled=false;
|
giuliomoro@53
|
39 return false;
|
giuliomoro@53
|
40 }
|
giuliomoro@44
|
41 server.sin_port=htons(port);
|
giuliomoro@44
|
42 if (bind(inSocket,(struct sockaddr *)&server,length)<0){
|
giuliomoro@44
|
43 enabled=false;
|
giuliomoro@44
|
44 return false;
|
giuliomoro@44
|
45 }
|
giuliomoro@53
|
46 enabled=true;
|
giuliomoro@44
|
47 return true;
|
giuliomoro@44
|
48 };
|
giuliomoro@121
|
49 int UdpServer::waitUntilReady(bool readyForReading, int timeoutMsecs){
|
giuliomoro@121
|
50 // If the socket is ready on return, this returns 1. If it times-out before the socket becomes ready, it returns 0. If an error occurs, it returns -1.
|
giuliomoro@121
|
51 if(enabled==false)
|
giuliomoro@121
|
52 return -1;
|
giuliomoro@121
|
53 if(timeoutMsecs<0)
|
giuliomoro@121
|
54 return select(inSocket+1, &stReadFDS, NULL, NULL, NULL); //calling this with a NULL timeout will block indefinitely
|
giuliomoro@121
|
55 FD_ZERO(&stReadFDS);
|
giuliomoro@121
|
56 FD_SET(inSocket, &stReadFDS);
|
giuliomoro@121
|
57 if(timeoutMsecs>=1000){
|
giuliomoro@121
|
58 float timeOutSecs=timeoutMsecs*0.001;
|
giuliomoro@121
|
59 stTimeOut.tv_sec=(long int)timeOutSecs;
|
giuliomoro@121
|
60 timeOutSecs-=(int)timeOutSecs;
|
giuliomoro@121
|
61 stTimeOut.tv_usec=(long int)(timeOutSecs*1000000);
|
giuliomoro@121
|
62 } else //faster!
|
giuliomoro@121
|
63 stTimeOut.tv_usec=(long int)timeoutMsecs*1000;
|
giuliomoro@121
|
64 int descriptorReady= select(inSocket+1, &stReadFDS, NULL, NULL, &stTimeOut);
|
giuliomoro@121
|
65 return descriptorReady>0? 1 : descriptorReady;
|
giuliomoro@121
|
66 }
|
giuliomoro@121
|
67
|
giuliomoro@44
|
68 int UdpServer::read(//Returns the number of bytes read, or -1 if there was an error.
|
giuliomoro@44
|
69 void *destBuffer,
|
giuliomoro@121
|
70 int maxBytesToRead,
|
giuliomoro@121
|
71 bool blockUntilSpecifiedAmountHasArrived)
|
giuliomoro@121
|
72 {
|
giuliomoro@44
|
73 if(enabled==false)
|
giuliomoro@44
|
74 return -1;
|
giuliomoro@44
|
75 FD_ZERO(&stReadFDS);
|
giuliomoro@44
|
76 FD_SET(inSocket, &stReadFDS);
|
giuliomoro@121
|
77 int descriptorReady= select(inSocket+1, &stReadFDS, NULL, NULL, &stZeroTimeOut);
|
giuliomoro@44
|
78 if(descriptorReady<0){ //an error occurred
|
giuliomoro@44
|
79 return -1;
|
giuliomoro@44
|
80 }
|
giuliomoro@44
|
81 int numberOfBytes=0;
|
giuliomoro@121
|
82 // do
|
giuliomoro@121
|
83 {
|
giuliomoro@121
|
84 if (FD_ISSET(inSocket, &stReadFDS))
|
giuliomoro@121
|
85 {
|
giuliomoro@121
|
86 // numberOfBytes=recvfrom(inSocket,destBuffer,maxBytesToRead,0,(struct sockaddr *)&from,&fromLength);
|
giuliomoro@121
|
87 numberOfBytes+=recv(inSocket,destBuffer,maxBytesToRead-numberOfBytes,0);
|
giuliomoro@121
|
88 if(numberOfBytes<0)
|
giuliomoro@121
|
89 return -1;
|
giuliomoro@121
|
90 }
|
giuliomoro@44
|
91 }
|
giuliomoro@121
|
92 // while (blockUntilSpecifiedAmountHasArrived && numberOfBytes==maxBytesToRead);
|
giuliomoro@44
|
93 return numberOfBytes;
|
giuliomoro@121
|
94 }
|
giuliomoro@44
|
95 int UdpServer::emptySocket(){
|
giuliomoro@44
|
96 return emptySocket(0);
|
giuliomoro@44
|
97 }
|
giuliomoro@44
|
98 int UdpServer::emptySocket(int maxBytes){//discards up to maxBytes from the socket. Returns the number of bytes discarded.
|
giuliomoro@44
|
99 if(wasteBuffer==NULL)
|
giuliomoro@44
|
100 return -1;
|
giuliomoro@44
|
101 int numberOfBytes=0;
|
giuliomoro@121
|
102 while(int n=read(wasteBuffer, wasteBufferSize, false)){// calls the read function until it does not return any more bytes (i.e.: socket is empty)
|
giuliomoro@44
|
103 if(n>0)
|
giuliomoro@44
|
104 numberOfBytes+=n;
|
giuliomoro@44
|
105 if(n<0)
|
giuliomoro@44
|
106 return -1;
|
giuliomoro@44
|
107 if(maxBytes>0 && numberOfBytes>=maxBytes)
|
giuliomoro@44
|
108 break;
|
giuliomoro@44
|
109 };
|
giuliomoro@44
|
110 return numberOfBytes;
|
giuliomoro@44
|
111 }
|
giuliomoro@44
|
112 void* UdpServer::getWaste(){ //returns the last datagram retrieved by emptySocket()
|
giuliomoro@44
|
113 return wasteBuffer;
|
giuliomoro@44
|
114 }
|
giuliomoro@44
|
115
|