annotate core/UdpServer.cpp @ 121:2197435e8fb4 scope-refactoring

UdpServer : read and waitUntilReady are now somehow Juce-compliant
author Giulio Moro <giuliomoro@yahoo.it>
date Sat, 22 Aug 2015 01:20:35 +0100
parents d3f869b98147
children 23137a333c93
rev   line source
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