annotate core/ClockSync.cpp @ 149:134bff10e561 ClockSync

Added simple one-variable one-measurement Kalman filter, Pid controller(which output is not used). Virtual clock is now much more precise and reactive for period. Still it is lagging behind a bit on the overall offset.
author Giulio Moro <giuliomoro@yahoo.it>
date Mon, 21 Sep 2015 03:12:21 +0100
parents 55c1e591cb2e
children e9c9404e3d1f
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@141 13 resetTs();
giuliomoro@141 14 receiveLoopSleepUs=100;
giuliomoro@141 15 receiveLoopTimeout=1e5;
giuliomoro@141 16 movingAverage.setLength(31);
giuliomoro@141 17 expectedClockSyncType=isSlave() ? kSync : kNone;
giuliomoro@141 18 }
giuliomoro@141 19 void ClockSync::resetTs(){
giuliomoro@135 20 T1=-1;
giuliomoro@135 21 T1p=-1;
giuliomoro@135 22 T2=-1;
giuliomoro@135 23 T2p=-1;
giuliomoro@141 24 }
giuliomoro@141 25 bool ClockSync::areTsValid(){
giuliomoro@141 26 return T1>0 && T1p>0 && T2>0 && T2p>0;
giuliomoro@135 27 }
giuliomoro@135 28 ClockSync::ClockSync(bool thisIsSlave, int aPort, VirtualClock &aVirtualClock){
giuliomoro@135 29 init(thisIsSlave, aPort, aVirtualClock);
giuliomoro@135 30 }
giuliomoro@135 31 void* ClockSync::getBuffer(){
giuliomoro@135 32 return buffer;
giuliomoro@135 33 }
giuliomoro@135 34 bool ClockSync::isSlave(){
giuliomoro@135 35 return slave;
giuliomoro@135 36 }
giuliomoro@135 37 bool ClockSync::isMaster(){
giuliomoro@135 38 return !slave;
giuliomoro@135 39 }
giuliomoro@135 40 int ClockSync::getType(){
giuliomoro@135 41 return ((int*)buffer)[0];
giuliomoro@135 42 }
giuliomoro@135 43 myClock_t ClockSync::getTimestamp(){
giuliomoro@135 44 return *((myClock_t*)((char*)buffer+sizeof(int)));
giuliomoro@135 45 }
giuliomoro@135 46 void ClockSync::setType(int clockSyncType){
giuliomoro@135 47 ((int*)buffer)[0]=clockSyncType;
giuliomoro@135 48 }
giuliomoro@135 49 void ClockSync::setTimestamp(myClock_t timestamp){
giuliomoro@139 50 // printf("setting timestamp: %lld\n", timestamp);
giuliomoro@135 51 ((myClock_t*)((char*)buffer+sizeof(int)))[0]=timestamp;
giuliomoro@135 52 }
giuliomoro@135 53 void ClockSync::print(){
giuliomoro@135 54 //printf("type: %d, timestamp: %lld\n",*((int*)buffer),*((myClock_t*)(((int*)buffer)+1)));
giuliomoro@135 55 }
giuliomoro@135 56 void ClockSync::setPort(int aPort){
giuliomoro@135 57 if(aPort>0){
giuliomoro@135 58 int inPort=isSlave() ? aPort : aPort+1;
giuliomoro@135 59 int outPort=isSlave() ? aPort+1: aPort;
giuliomoro@135 60 server.bindToPort(inPort);
giuliomoro@135 61 client.setPort(outPort);
giuliomoro@135 62 //printf("Receiving on port %d\n", inPort);
giuliomoro@135 63 //printf("Sending to port %d\n", outPort);
giuliomoro@135 64 }
giuliomoro@135 65 }
giuliomoro@135 66 /**
giuliomoro@135 67 * sends a clockSync without blocking, checks results and returns the timestamp
giuliomoro@135 68 * immediately after the clockSync has been sent or -1 if there was an error or timeout expired.
giuliomoro@135 69 */
giuliomoro@135 70 myClock_t ClockSync::send(){
giuliomoro@135 71 // print();
giuliomoro@135 72 int ret;
giuliomoro@149 73 ret=client.waitUntilReady(false, isSlave() ? 110 : 5);
giuliomoro@135 74 if(ret<=0){ //don't retry
giuliomoro@135 75 return -1;
giuliomoro@135 76 }
giuliomoro@135 77 ret=client.send(buffer, bufferLength);
giuliomoro@135 78 myClock_t timestamp=(myClock_t)virtualClock->getNow();
giuliomoro@135 79 if(ret<0){
giuliomoro@135 80 //if there was an error while sending, don't retry
giuliomoro@135 81 return -1;
giuliomoro@135 82 }
giuliomoro@135 83 return timestamp; //get the accurate time *after* you sent the sync clockSync
giuliomoro@135 84 }
giuliomoro@135 85 /**
giuliomoro@135 86 * receives a clockSync without blocking, checks results and returns the timestamp
giuliomoro@135 87 * immediately after the clockSync has been received, or -1 if there was an error
giuliomoro@135 88 * or 0 if timeout expired.
giuliomoro@135 89 */
giuliomoro@135 90 myClock_t ClockSync::receive(){
giuliomoro@135 91 int ret;
giuliomoro@135 92 ret=server.waitUntilReady(true, 0);
giuliomoro@135 93 if(ret<=0){ //don't retry
giuliomoro@135 94 return 0;
giuliomoro@135 95 }
giuliomoro@135 96 ret=server.read(buffer, bufferLength, false);
giuliomoro@135 97 myClock_t timestamp=(myClock_t)virtualClock->getNow();
giuliomoro@135 98 if(timestamp==0){
giuliomoro@135 99 //printf("The virtualClock period is <=0\n");
giuliomoro@135 100 return -3;
giuliomoro@135 101 }
giuliomoro@135 102 if(ret==-1){
giuliomoro@135 103 //if there was an error while receiving, don't retry
giuliomoro@135 104 return -1;
giuliomoro@135 105 }
giuliomoro@135 106 if(ret!=bufferLength){
giuliomoro@135 107 //printf("Received a clockSync of the wrong size: %d\n", ret);
giuliomoro@135 108 return -2;
giuliomoro@135 109 }
giuliomoro@135 110 // print();
giuliomoro@135 111 return timestamp; //get the accurate time *after* you sent the sync clockSync
giuliomoro@135 112 }
giuliomoro@135 113
giuliomoro@135 114 int ClockSync::masterSendSync(){
giuliomoro@135 115 //let's send a sync clockSync!
giuliomoro@135 116 //printf("Sending a sync clockSync\n");
giuliomoro@135 117 setType(kSync);
giuliomoro@135 118 setTimestamp(-1);//do not care about sending the timestamp, a more accurate one will be sent in the follow up
giuliomoro@135 119 localTimestamp=send();
giuliomoro@135 120 if(localTimestamp<0){
giuliomoro@135 121 //printf("Could not send sync clockSync\n");
giuliomoro@135 122 return -1; //error, don't retry, start over
giuliomoro@135 123 }
giuliomoro@135 124 //let's send a followUp
giuliomoro@135 125 //printf("sent a sync clockSync\n");
giuliomoro@135 126 setType(kFollowUp);
giuliomoro@135 127 setTimestamp(localTimestamp);
giuliomoro@135 128 if(localTimestamp<0){
giuliomoro@135 129 //printf("Could not send followup clockSync\n");
giuliomoro@135 130 return -2; //error, don't retry, start over
giuliomoro@135 131 }
giuliomoro@135 132 int ret=send();
giuliomoro@135 133 if(ret<0){
giuliomoro@135 134 //printf("Error while sending followup\n");
giuliomoro@135 135 return -3;
giuliomoro@135 136 }
giuliomoro@135 137 //printf("sent a followUp clockSync\n");
giuliomoro@135 138 expectedClockSyncType=kDelayReq;
giuliomoro@135 139 return 1;
giuliomoro@135 140 }
giuliomoro@135 141 #ifdef USE_JUCE
giuliomoro@135 142 #define NOTGSHOULDSTOP 1
giuliomoro@135 143 #else
giuliomoro@135 144 extern bool gShouldStop;
giuliomoro@135 145 #define NOTGSHOULDSTOP (!gShouldStop)
giuliomoro@135 146 #endif /* USE_JUCE */
giuliomoro@135 147 int ClockSync::receiveLoop(){
giuliomoro@135 148 int receiveLoopElapsed=0;
giuliomoro@135 149 while( NOTGSHOULDSTOP && (isSlave() || (receiveLoopElapsed<receiveLoopTimeout))){ //when slave, does not timeout!
giuliomoro@135 150 receiveLoopElapsed+=receiveLoopSleepUs;
giuliomoro@135 151 usleep(receiveLoopSleepUs); //how often to check for new clockSyncs;
giuliomoro@135 152 // //printf("waiting for clockSyncs\n");
giuliomoro@135 153 localTimestamp=receive();
giuliomoro@135 154 if(localTimestamp<=0){
giuliomoro@135 155 if(localTimestamp==0){
giuliomoro@139 156 // printf("Socket not ready to be read: %lld\n", localTimestamp);
giuliomoro@135 157 }
giuliomoro@135 158 else if(localTimestamp==-1){
giuliomoro@139 159 printf("Error while receiving: %lld\n", localTimestamp);
giuliomoro@135 160 }
giuliomoro@135 161 else if(localTimestamp==-2){
giuliomoro@139 162 printf("Wrong size of the received clockSync: %lld\n", localTimestamp);
giuliomoro@135 163 }
giuliomoro@135 164 continue ; //keep waiting
giuliomoro@135 165 }
giuliomoro@135 166 clockSyncType=getType();
giuliomoro@135 167 clockSyncTimestamp=getTimestamp();
giuliomoro@135 168 if(clockSyncType!=expectedClockSyncType){
giuliomoro@135 169 //printf("Wrong clockSync type: %d, expected: %d\n",clockSyncType, expectedClockSyncType);
giuliomoro@135 170 return -2; //start over
giuliomoro@135 171 }
giuliomoro@139 172 // printf("Received clockSync type: %d, clockSyncTimestamp: %lld\n", clockSyncType, clockSyncTimestamp);
giuliomoro@135 173 if(isSlave()==true){
giuliomoro@135 174 int ret=slaveHandleMessage();
giuliomoro@135 175 if(ret==1 && clockSyncType==kDelayResp){ //we are done, end of a cycle!
giuliomoro@135 176 return 1;
giuliomoro@135 177 } else if (ret!=1) {
giuliomoro@135 178 return -1; //
giuliomoro@135 179 } else {
giuliomoro@135 180 continue;
giuliomoro@135 181 }
giuliomoro@135 182 }
giuliomoro@135 183 if(isMaster()==true){ //this is master
giuliomoro@135 184 int ret=masterHandleMessage();
giuliomoro@135 185 if(ret==1 && clockSyncType==kDelayReq){ //we are done, end of a cycle!
giuliomoro@135 186 return 1;
giuliomoro@135 187 } else {
giuliomoro@135 188 return -2; //we are done but something was wrong
giuliomoro@135 189 }
giuliomoro@135 190 }
giuliomoro@135 191 }
giuliomoro@135 192 //printf("Receive loop timeout\n");
giuliomoro@135 193 return -1;
giuliomoro@135 194 }
giuliomoro@135 195
giuliomoro@135 196 int ClockSync::slaveHandleMessage(){
giuliomoro@135 197 switch(clockSyncType){
giuliomoro@135 198 case kSync: //the clockSync timestamp is meaningless, the localTimestamp is when kSync was received
giuliomoro@141 199 resetTs();
giuliomoro@135 200 T1p=localTimestamp;
giuliomoro@135 201 expectedClockSyncType=kFollowUp;
giuliomoro@135 202 break;
giuliomoro@135 203 case kFollowUp: //the clockSyncTimestamp is the time when kSync was sent, the localTimestamp is meaningless
giuliomoro@135 204 T1=clockSyncTimestamp;
giuliomoro@135 205 //send delayReq
giuliomoro@135 206 setType(kDelayReq);
giuliomoro@135 207 setTimestamp(-1);
giuliomoro@135 208 T2=send();
giuliomoro@135 209 if(T2<0){
giuliomoro@135 210 //printf("Error while sending delayReq\n");
giuliomoro@135 211 return -1;
giuliomoro@135 212 }
giuliomoro@135 213 expectedClockSyncType=kDelayResp;
giuliomoro@135 214 break;
giuliomoro@135 215 case kDelayResp: {//the clockSyncTimestamp is the instant when the master received the kDelayResp clockSync, the localTimestamp is meaningless
giuliomoro@135 216 T2p=clockSyncTimestamp;
giuliomoro@135 217 //TODO: evaluate things
giuliomoro@135 218 double offset=(T1p-T1-T2p+T2)/2.0d;
giuliomoro@141 219 if(areTsValid()){
giuliomoro@141 220 processOffset(offset);
giuliomoro@141 221
giuliomoro@141 222 /*
giuliomoro@141 223 static int calls=0;
giuliomoro@141 224 static double referenceOffset=0;
giuliomoro@141 225
giuliomoro@141 226 if(calls<100){ // start by averaging everything
giuliomoro@141 227 movingAverage.add(offset);
giuliomoro@141 228 } else { //once we get an estimate, start discarding outliers
giuliomoro@141 229 float maxOffsetDeviation=20;
giuliomoro@141 230 float deviation=fabsf(movingAverage.getAverage()-offset);
giuliomoro@141 231 if(deviation<maxOffsetDeviation){
giuliomoro@141 232 movingAverage.add(offset);
giuliomoro@141 233 printf("w(end+1)=%4.1f;\n", movingAverage.getAverage());
giuliomoro@141 234 } else {
giuliomoro@141 235 // printf("Out of range: %f \n", deviation);
giuliomoro@141 236 }
giuliomoro@141 237 }
giuliomoro@141 238 printf("offset(end+1)=%f;\n", offset);
giuliomoro@141 239 if (calls==100){
giuliomoro@141 240 // printf("Update reference\n");
giuliomoro@141 241 referenceOffset=movingAverage.getAverage();
giuliomoro@141 242 } else if (calls==200){
giuliomoro@141 243 calls=99;
giuliomoro@141 244 }
giuliomoro@141 245 calls++;
giuliomoro@141 246 */
giuliomoro@141 247
giuliomoro@141 248 // printf("%lld, %lld, %lld, %lld \n", T1, T1p, T2, T2p);
giuliomoro@141 249 // if(T2-T1p<0){
giuliomoro@141 250 // printf("Negative: %lld, %lld, %lld, %lld \n", T1, T1p, T2, T2p);
giuliomoro@141 251 // }
giuliomoro@141 252 } else {
giuliomoro@141 253 printf("InvalidTs:\n %lld, %lld, %lld, %lld \n", T1, T1p, T2, T2p);
giuliomoro@141 254 }
giuliomoro@135 255 expectedClockSyncType=kSync; //end of the cycle, wait for next sync.
giuliomoro@135 256 break;
giuliomoro@135 257 }
giuliomoro@135 258 default:
giuliomoro@135 259 //printf("Unexpected message type\n"); // we should never get here
giuliomoro@135 260 return -1;
giuliomoro@135 261 }
giuliomoro@135 262 return 1;
giuliomoro@135 263 }
giuliomoro@141 264 void ClockSync::processOffset(double offset){
giuliomoro@141 265 static int calls=0;
giuliomoro@141 266 // TODO: change the flow control below so that it can happen multiple times
giuliomoro@141 267 //(base it upon the length of movingAverage rather than the number of calls)
giuliomoro@141 268 if(calls<10) { //get an initial guess
giuliomoro@141 269 movingAverage.add(offset);
giuliomoro@141 270 // printf("-----------OFFSET IS : %04.4f samples, average: %04.4f samples\n",
giuliomoro@141 271 // offset, movingAverage.getAverage());
giuliomoro@141 272 } else if (calls==10){ //then compensate for initial offset
giuliomoro@149 273 printf("compensating for offset: %f\n", offset);
giuliomoro@141 274 virtualClock->addOffset(movingAverage.getAverage());
giuliomoro@141 275 movingAverage.reset();
giuliomoro@141 276 } else if (calls>=10){ //use IIR filter from now on
giuliomoro@141 277 //filter coefficients obtained from Matlab : [B,A]=butter(2,0.005);
giuliomoro@141 278 // static float B[3]={6.10061787580662e-05, 0.000122012357516132, 6.10061787580662e-05};
giuliomoro@141 279 // static float A[3]={1, -1.97778648377676, 0.978030508491796};
giuliomoro@141 280 static float B[3]={6.10061787580662e-05, 0.000122012357516132, 6.10061787580662e-05};
giuliomoro@141 281 static float A[3]={1, -1.97778648377676, 0.978030508491796};
giuliomoro@141 282 static float pastOut[3]={0,0,0};
giuliomoro@141 283 static float pastIn[3]={0,0,0};
giuliomoro@141 284 float in=offset;
giuliomoro@141 285 float out= -pastOut[1]*A[1] -pastOut[2]*A[2] +in*B[0] +pastIn[1]*B[1] +pastIn[2]*B[2];
giuliomoro@141 286 pastOut[2]=pastOut[1];
giuliomoro@141 287 pastOut[1]=out;
giuliomoro@141 288 pastIn[2]=pastIn[1];
giuliomoro@141 289 pastIn[1]=in;
giuliomoro@141 290 offset=out;
giuliomoro@141 291 static float maxOffset=0;
giuliomoro@149 292 if(calls > 0 ) {
giuliomoro@149 293 maxOffset=fabsf(offset) > fabsf(maxOffset) ? offset : maxOffset;
giuliomoro@149 294 pid.setError(offset);
giuliomoro@149 295 float correction=pid.getOutput();
giuliomoro@149 296 static float oldSamplingRate=44100;
giuliomoro@149 297 printf("%10.3f, %10.3f, %10.3f, %10.3f, %10.3f, %10.3f\n", in, offset, offset-pastOut[2], maxOffset, correction, oldSamplingRate); //unfiltered, filtered
giuliomoro@149 298 // if(fabsf(offset)>3 && calls>30){
giuliomoro@149 299 //TODO: correct for offset
giuliomoro@149 300 // float targetSamplingRate=offset>0 ? 44095 : 44105;
giuliomoro@149 301 float targetSamplingRate = oldSamplingRate - correction;
giuliomoro@149 302 #ifndef USE_JUCE
giuliomoro@149 303 // if(oldSamplingRate != targetSamplingRate)
giuliomoro@149 304 // gAudioCodec->setAudioSamplingRate(targetSamplingRate);
giuliomoro@149 305 #endif
giuliomoro@149 306 oldSamplingRate=targetSamplingRate;
giuliomoro@149 307 // pastOut[1]=pastOut[2]=pastIn[1]=pastIn[2]=offset;
giuliomoro@149 308 // printf("------setAudioSmplingRate to %f\n", targetSamplingRate);
giuliomoro@149 309 // }
giuliomoro@141 310 }
giuliomoro@141 311 }
giuliomoro@141 312 calls++;
giuliomoro@141 313 }
giuliomoro@135 314 int ClockSync::masterHandleMessage(){
giuliomoro@135 315 switch(clockSyncType){
giuliomoro@135 316 case kDelayReq:
giuliomoro@135 317 //send kDelayResp
giuliomoro@135 318 setType(kDelayResp);
giuliomoro@135 319 setTimestamp(localTimestamp);
giuliomoro@135 320 send();
giuliomoro@135 321 expectedClockSyncType=kNone;
giuliomoro@141 322 return 1;
giuliomoro@135 323 break;
giuliomoro@141 324 default:
giuliomoro@141 325 return -1;
giuliomoro@135 326 }
giuliomoro@135 327 }
giuliomoro@135 328
giuliomoro@135 329 int ClockSync::sendReceiveLoop(){
giuliomoro@135 330 if(isSlave()==true){
giuliomoro@135 331 //printf("Waiting for a sync clockSync\n");
giuliomoro@135 332 } else { //if this is master
giuliomoro@135 333 usleep(100000); //this times (roughly) how often sync clockSyncs are being sent.
giuliomoro@135 334 int ret=masterSendSync();
giuliomoro@135 335 if(ret<=0)
giuliomoro@135 336 return -1;
giuliomoro@135 337 }
giuliomoro@135 338 int ret=receiveLoop();
giuliomoro@135 339 if(ret<=0)
giuliomoro@135 340 return -2;
giuliomoro@135 341 return 1;
giuliomoro@135 342 }
giuliomoro@135 343
giuliomoro@135 344