annotate core/ClockSync.cpp @ 139:4e2dd3eb1d28 ClockSync

The reported offset is now meaningful. The whole thing is waaay too jittery.
author Giulio Moro <giuliomoro@yahoo.it>
date Sun, 13 Sep 2015 21:34:47 +0100
parents e77e2e712fbc
children 44d07fa9bd03
rev   line source
giuliomoro@135 1 #include "ClockSync.h"
giuliomoro@135 2
giuliomoro@135 3 void ClockSync::setVirtualClock(VirtualClock &aVirtualClock){
giuliomoro@135 4 virtualClock=&aVirtualClock;
giuliomoro@135 5 }
giuliomoro@135 6 void ClockSync::init(bool thisIsSlave, int aPort, VirtualClock &aVirtualClock){
giuliomoro@135 7 setVirtualClock(aVirtualClock);
giuliomoro@135 8 slave=thisIsSlave;
giuliomoro@135 9 setPort(aPort);
giuliomoro@135 10 // isSlave() ? client.setServer("127.0.0.1") : client.setServer("127.0.0.1");
giuliomoro@135 11 isSlave() ? client.setServer("192.168.7.1") : client.setServer("192.168.7.2");
giuliomoro@135 12 bufferLength=kSyncMessageLength;
giuliomoro@135 13 T1=-1;
giuliomoro@135 14 T1p=-1;
giuliomoro@135 15 T2=-1;
giuliomoro@135 16 T2p=-1;
giuliomoro@139 17 receiveLoopSleepUs=100;
giuliomoro@135 18 receiveLoopTimeout=1e5;
giuliomoro@139 19 movingAverage.setLength(31);
giuliomoro@135 20 expectedClockSyncType=isSlave() ? kSync : kNone;
giuliomoro@135 21 }
giuliomoro@135 22 ClockSync::ClockSync(bool thisIsSlave, int aPort, VirtualClock &aVirtualClock){
giuliomoro@135 23 init(thisIsSlave, aPort, aVirtualClock);
giuliomoro@135 24 }
giuliomoro@135 25 void* ClockSync::getBuffer(){
giuliomoro@135 26 return buffer;
giuliomoro@135 27 }
giuliomoro@135 28 bool ClockSync::isSlave(){
giuliomoro@135 29 return slave;
giuliomoro@135 30 }
giuliomoro@135 31 bool ClockSync::isMaster(){
giuliomoro@135 32 return !slave;
giuliomoro@135 33 }
giuliomoro@135 34 int ClockSync::getType(){
giuliomoro@135 35 return ((int*)buffer)[0];
giuliomoro@135 36 }
giuliomoro@135 37 myClock_t ClockSync::getTimestamp(){
giuliomoro@135 38 return *((myClock_t*)((char*)buffer+sizeof(int)));
giuliomoro@135 39 }
giuliomoro@135 40 void ClockSync::setType(int clockSyncType){
giuliomoro@135 41 ((int*)buffer)[0]=clockSyncType;
giuliomoro@135 42 }
giuliomoro@135 43 void ClockSync::setTimestamp(myClock_t timestamp){
giuliomoro@139 44 // printf("setting timestamp: %lld\n", timestamp);
giuliomoro@135 45 ((myClock_t*)((char*)buffer+sizeof(int)))[0]=timestamp;
giuliomoro@135 46 }
giuliomoro@135 47 void ClockSync::print(){
giuliomoro@135 48 //printf("type: %d, timestamp: %lld\n",*((int*)buffer),*((myClock_t*)(((int*)buffer)+1)));
giuliomoro@135 49 }
giuliomoro@135 50 void ClockSync::setPort(int aPort){
giuliomoro@135 51 if(aPort>0){
giuliomoro@135 52 int inPort=isSlave() ? aPort : aPort+1;
giuliomoro@135 53 int outPort=isSlave() ? aPort+1: aPort;
giuliomoro@135 54 server.bindToPort(inPort);
giuliomoro@135 55 client.setPort(outPort);
giuliomoro@135 56 //printf("Receiving on port %d\n", inPort);
giuliomoro@135 57 //printf("Sending to port %d\n", outPort);
giuliomoro@135 58 }
giuliomoro@135 59 }
giuliomoro@135 60 /**
giuliomoro@135 61 * sends a clockSync without blocking, checks results and returns the timestamp
giuliomoro@135 62 * immediately after the clockSync has been sent or -1 if there was an error or timeout expired.
giuliomoro@135 63 */
giuliomoro@135 64 myClock_t ClockSync::send(){
giuliomoro@135 65 // print();
giuliomoro@135 66 int ret;
giuliomoro@135 67 ret=client.waitUntilReady(false, 0);
giuliomoro@135 68 if(ret<=0){ //don't retry
giuliomoro@135 69 return -1;
giuliomoro@135 70 }
giuliomoro@135 71 ret=client.send(buffer, bufferLength);
giuliomoro@135 72 myClock_t timestamp=(myClock_t)virtualClock->getNow();
giuliomoro@135 73 if(ret<0){
giuliomoro@135 74 //if there was an error while sending, don't retry
giuliomoro@135 75 return -1;
giuliomoro@135 76 }
giuliomoro@135 77 return timestamp; //get the accurate time *after* you sent the sync clockSync
giuliomoro@135 78 }
giuliomoro@135 79 /**
giuliomoro@135 80 * receives a clockSync without blocking, checks results and returns the timestamp
giuliomoro@135 81 * immediately after the clockSync has been received, or -1 if there was an error
giuliomoro@135 82 * or 0 if timeout expired.
giuliomoro@135 83 */
giuliomoro@135 84 myClock_t ClockSync::receive(){
giuliomoro@135 85 int ret;
giuliomoro@135 86 ret=server.waitUntilReady(true, 0);
giuliomoro@135 87 if(ret<=0){ //don't retry
giuliomoro@135 88 return 0;
giuliomoro@135 89 }
giuliomoro@135 90 ret=server.read(buffer, bufferLength, false);
giuliomoro@135 91 myClock_t timestamp=(myClock_t)virtualClock->getNow();
giuliomoro@135 92 if(timestamp==0){
giuliomoro@135 93 //printf("The virtualClock period is <=0\n");
giuliomoro@135 94 return -3;
giuliomoro@135 95 }
giuliomoro@135 96 if(ret==-1){
giuliomoro@135 97 //if there was an error while receiving, don't retry
giuliomoro@135 98 return -1;
giuliomoro@135 99 }
giuliomoro@135 100 if(ret!=bufferLength){
giuliomoro@135 101 //printf("Received a clockSync of the wrong size: %d\n", ret);
giuliomoro@135 102 return -2;
giuliomoro@135 103 }
giuliomoro@135 104 // print();
giuliomoro@135 105 return timestamp; //get the accurate time *after* you sent the sync clockSync
giuliomoro@135 106 }
giuliomoro@135 107
giuliomoro@135 108 int ClockSync::masterSendSync(){
giuliomoro@135 109 //let's send a sync clockSync!
giuliomoro@135 110 //printf("Sending a sync clockSync\n");
giuliomoro@135 111 setType(kSync);
giuliomoro@135 112 setTimestamp(-1);//do not care about sending the timestamp, a more accurate one will be sent in the follow up
giuliomoro@135 113 localTimestamp=send();
giuliomoro@135 114 if(localTimestamp<0){
giuliomoro@135 115 //printf("Could not send sync clockSync\n");
giuliomoro@135 116 return -1; //error, don't retry, start over
giuliomoro@135 117 }
giuliomoro@135 118 //let's send a followUp
giuliomoro@135 119 //printf("sent a sync clockSync\n");
giuliomoro@135 120 setType(kFollowUp);
giuliomoro@135 121 setTimestamp(localTimestamp);
giuliomoro@135 122 if(localTimestamp<0){
giuliomoro@135 123 //printf("Could not send followup clockSync\n");
giuliomoro@135 124 return -2; //error, don't retry, start over
giuliomoro@135 125 }
giuliomoro@135 126 int ret=send();
giuliomoro@135 127 if(ret<0){
giuliomoro@135 128 //printf("Error while sending followup\n");
giuliomoro@135 129 return -3;
giuliomoro@135 130 }
giuliomoro@135 131 //printf("sent a followUp clockSync\n");
giuliomoro@135 132 expectedClockSyncType=kDelayReq;
giuliomoro@135 133 return 1;
giuliomoro@135 134 }
giuliomoro@135 135 #ifdef USE_JUCE
giuliomoro@135 136 #define NOTGSHOULDSTOP 1
giuliomoro@135 137 #else
giuliomoro@135 138 extern bool gShouldStop;
giuliomoro@135 139 #define NOTGSHOULDSTOP (!gShouldStop)
giuliomoro@135 140 #endif /* USE_JUCE */
giuliomoro@135 141 int ClockSync::receiveLoop(){
giuliomoro@135 142 int receiveLoopElapsed=0;
giuliomoro@135 143 while( NOTGSHOULDSTOP && (isSlave() || (receiveLoopElapsed<receiveLoopTimeout))){ //when slave, does not timeout!
giuliomoro@135 144 receiveLoopElapsed+=receiveLoopSleepUs;
giuliomoro@135 145 usleep(receiveLoopSleepUs); //how often to check for new clockSyncs;
giuliomoro@135 146 // //printf("waiting for clockSyncs\n");
giuliomoro@135 147 localTimestamp=receive();
giuliomoro@135 148 if(localTimestamp<=0){
giuliomoro@135 149 if(localTimestamp==0){
giuliomoro@139 150 // printf("Socket not ready to be read: %lld\n", localTimestamp);
giuliomoro@135 151 }
giuliomoro@135 152 else if(localTimestamp==-1){
giuliomoro@139 153 printf("Error while receiving: %lld\n", localTimestamp);
giuliomoro@135 154 }
giuliomoro@135 155 else if(localTimestamp==-2){
giuliomoro@139 156 printf("Wrong size of the received clockSync: %lld\n", localTimestamp);
giuliomoro@135 157 }
giuliomoro@135 158 continue ; //keep waiting
giuliomoro@135 159 }
giuliomoro@135 160 clockSyncType=getType();
giuliomoro@135 161 clockSyncTimestamp=getTimestamp();
giuliomoro@135 162 if(clockSyncType!=expectedClockSyncType){
giuliomoro@135 163 //printf("Wrong clockSync type: %d, expected: %d\n",clockSyncType, expectedClockSyncType);
giuliomoro@135 164 return -2; //start over
giuliomoro@135 165 }
giuliomoro@139 166 // printf("Received clockSync type: %d, clockSyncTimestamp: %lld\n", clockSyncType, clockSyncTimestamp);
giuliomoro@135 167 if(isSlave()==true){
giuliomoro@135 168 int ret=slaveHandleMessage();
giuliomoro@135 169 if(ret==1 && clockSyncType==kDelayResp){ //we are done, end of a cycle!
giuliomoro@135 170 return 1;
giuliomoro@135 171 } else if (ret!=1) {
giuliomoro@135 172 return -1; //
giuliomoro@135 173 } else {
giuliomoro@135 174 continue;
giuliomoro@135 175 }
giuliomoro@135 176 }
giuliomoro@135 177 if(isMaster()==true){ //this is master
giuliomoro@135 178 int ret=masterHandleMessage();
giuliomoro@135 179 if(ret==1 && clockSyncType==kDelayReq){ //we are done, end of a cycle!
giuliomoro@135 180 return 1;
giuliomoro@135 181 } else {
giuliomoro@135 182 return -2; //we are done but something was wrong
giuliomoro@135 183 }
giuliomoro@135 184 }
giuliomoro@135 185 }
giuliomoro@135 186 //printf("Receive loop timeout\n");
giuliomoro@135 187 return -1;
giuliomoro@135 188 }
giuliomoro@135 189
giuliomoro@135 190 int ClockSync::slaveHandleMessage(){
giuliomoro@135 191 switch(clockSyncType){
giuliomoro@135 192 case kSync: //the clockSync timestamp is meaningless, the localTimestamp is when kSync was received
giuliomoro@135 193 T1p=localTimestamp;
giuliomoro@135 194 expectedClockSyncType=kFollowUp;
giuliomoro@135 195 break;
giuliomoro@135 196 case kFollowUp: //the clockSyncTimestamp is the time when kSync was sent, the localTimestamp is meaningless
giuliomoro@135 197 T1=clockSyncTimestamp;
giuliomoro@135 198 //send delayReq
giuliomoro@135 199 setType(kDelayReq);
giuliomoro@135 200 setTimestamp(-1);
giuliomoro@135 201 T2=send();
giuliomoro@135 202 if(T2<0){
giuliomoro@135 203 //printf("Error while sending delayReq\n");
giuliomoro@135 204 return -1;
giuliomoro@135 205 }
giuliomoro@135 206 expectedClockSyncType=kDelayResp;
giuliomoro@135 207 break;
giuliomoro@135 208 case kDelayResp: {//the clockSyncTimestamp is the instant when the master received the kDelayResp clockSync, the localTimestamp is meaningless
giuliomoro@135 209 T2p=clockSyncTimestamp;
giuliomoro@135 210 //TODO: evaluate things
giuliomoro@135 211 double offset=(T1p-T1-T2p+T2)/2.0d;
giuliomoro@139 212 printf("-----------OFFSET IS : %04.4f seconds, average: %04.4f seconds\n",
giuliomoro@139 213 offset/44100.f, movingAverage.add(offset)/44100.f);
giuliomoro@135 214 expectedClockSyncType=kSync; //end of the cycle, wait for next sync.
giuliomoro@135 215 break;
giuliomoro@135 216 }
giuliomoro@135 217 default:
giuliomoro@135 218 //printf("Unexpected message type\n"); // we should never get here
giuliomoro@135 219 return -1;
giuliomoro@135 220 }
giuliomoro@135 221 return 1;
giuliomoro@135 222 }
giuliomoro@135 223
giuliomoro@135 224 int ClockSync::masterHandleMessage(){
giuliomoro@135 225 switch(clockSyncType){
giuliomoro@135 226 case kDelayReq:
giuliomoro@135 227 //TODO: do something with it
giuliomoro@135 228 //send kDelayResp
giuliomoro@135 229 setType(kDelayResp);
giuliomoro@135 230 setTimestamp(localTimestamp);
giuliomoro@135 231 send();
giuliomoro@135 232 expectedClockSyncType=kNone;
giuliomoro@135 233 break;
giuliomoro@135 234 }
giuliomoro@135 235 }
giuliomoro@135 236
giuliomoro@135 237 int ClockSync::sendReceiveLoop(){
giuliomoro@135 238 if(isSlave()==true){
giuliomoro@135 239 //printf("Waiting for a sync clockSync\n");
giuliomoro@135 240 } else { //if this is master
giuliomoro@135 241 usleep(100000); //this times (roughly) how often sync clockSyncs are being sent.
giuliomoro@135 242 int ret=masterSendSync();
giuliomoro@135 243 if(ret<=0)
giuliomoro@135 244 return -1;
giuliomoro@135 245 }
giuliomoro@135 246 int ret=receiveLoop();
giuliomoro@135 247 if(ret<=0)
giuliomoro@135 248 return -2;
giuliomoro@135 249 return 1;
giuliomoro@135 250 }
giuliomoro@135 251
giuliomoro@135 252