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
|