Mercurial > hg > beaglert
view core/UdpServer.cpp @ 125:850a4a9bd832 scope-refactoring
Added ifdefs and unified the code with udpioplugin ... the latter has not been tested (or committed). TODO: still it hangs after ctrl-c BeagleRT (auxiliary tasks do not terminate). TODO: sometimes you can hear dropouts in the transmission. Maybe it is due to pointer drifting. Rebooting BBB fixes/affects this issue.
author | Giulio Moro <giuliomoro@yahoo.it> |
---|---|
date | Sat, 22 Aug 2015 02:53:36 +0100 |
parents | 23137a333c93 |
children | 719119fb2905 |
line wrap: on
line source
/* * udpServer.cpp * * Created on: 19 May 2015 * Author: giulio moro */ #include "../include/UdpServer.h" UdpServer::UdpServer(int aPort){ init(aPort); }; UdpServer::UdpServer(){ init(0); } UdpServer::~UdpServer(){ printf("Close the socket\n"); shutdown(inSocket, 0); //Stop receiving data for this socket. If further data arrives, reject it. //TODO: unbind from port. AFAIK, this involves closing the socket, therefore creating the socket should become part of bindToPort }; bool UdpServer::init(int aPort){ enabled=true; stZeroTimeOut.tv_sec = 0; //set timeout to 0 stZeroTimeOut.tv_usec = 0; inSocket=socket(AF_INET, SOCK_DGRAM, 0); if (inSocket < 0){ enabled=false; } length = sizeof(server); server.sin_family=AF_INET; server.sin_addr.s_addr=INADDR_ANY; enabled=bindToPort(aPort); wasteBufferSize=2048; wasteBuffer=malloc(wasteBufferSize); return enabled; } bool UdpServer::bindToPort(int aPort){ port=aPort; if(port<1){ enabled=false; return false; } server.sin_port=htons(port); if (bind(inSocket,(struct sockaddr *)&server,length)<0){ enabled=false; return false; } enabled=true; return true; }; int UdpServer::waitUntilReady(bool readyForReading, int timeoutMsecs){ // 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. if(enabled==false) return -1; if(timeoutMsecs<0) return select(inSocket+1, &stReadFDS, NULL, NULL, NULL); //calling this with a NULL timeout will block indefinitely FD_ZERO(&stReadFDS); FD_SET(inSocket, &stReadFDS); if(timeoutMsecs>=1000){ float timeOutSecs=timeoutMsecs*0.001; stTimeOut.tv_sec=(long int)timeOutSecs; timeOutSecs-=(int)timeOutSecs; stTimeOut.tv_usec=(long int)(timeOutSecs*1000000); } else //faster! stTimeOut.tv_usec=(long int)timeoutMsecs*1000; int descriptorReady= select(inSocket+1, &stReadFDS, NULL, NULL, &stTimeOut); return descriptorReady>0? 1 : descriptorReady; } int UdpServer::read(//Returns the number of bytes read, or -1 if there was an error. void *destBuffer, int maxBytesToRead, bool blockUntilSpecifiedAmountHasArrived) { if(enabled==false) return -1; FD_ZERO(&stReadFDS); FD_SET(inSocket, &stReadFDS); int descriptorReady= select(inSocket+1, &stReadFDS, NULL, NULL, &stZeroTimeOut); if(descriptorReady<0){ //an error occurred return -1; } int numberOfBytes=0; // do { if (FD_ISSET(inSocket, &stReadFDS)) { // numberOfBytes=recvfrom(inSocket,destBuffer,maxBytesToRead,0,(struct sockaddr *)&from,&fromLength); numberOfBytes+=recv(inSocket,destBuffer,maxBytesToRead-numberOfBytes,0); if(numberOfBytes<0) return -1; } } // while (blockUntilSpecifiedAmountHasArrived && numberOfBytes==maxBytesToRead); return numberOfBytes; } int UdpServer::emptySocket(){ return emptySocket(0); } int UdpServer::emptySocket(int maxBytes){//discards up to maxBytes from the socket. Returns the number of bytes discarded. if(wasteBuffer==NULL) return -1; int numberOfBytes=0; while(int n=read(wasteBuffer, wasteBufferSize, false)){// calls the read function until it does not return any more bytes (i.e.: socket is empty) if(n>0) numberOfBytes+=n; if(n<0) return -1; if(maxBytes>0 && numberOfBytes>=maxBytes) break; }; return numberOfBytes; } void* UdpServer::getWaste(){ //returns the last datagram retrieved by emptySocket() return wasteBuffer; }