-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathmaster.c
246 lines (232 loc) · 6.08 KB
/
master.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
#include <stddef.h>
#include "ptp_common.h"
#include <limits.h>
#include <string.h>
#include <stdio.h>
/*
SLAVE
loop
if receive PTPM_START
send PTPM_START
receive PTP_TIMES => times
clocksync(times):
sync:
receive * (should be PTPM_SYNC else abort)
now=t2
send (PTPM_SYNC,t2)
delay:
receive * (should be PPTM_DELAY else abort)
now=t3
send (PPTM_DELAY,t3)
receive_packet * (should be PPTM_NEXT else abort)
else:
send PPTS_HELLO
*/
/*
MASTER
send PTPM_START
wait receive PTPM_START
send PTPM_TIMES #
wait receive PTPM_TIMES
clock:
loop times #
sync
send PTPM_SYNC
now=t1
receive_packet (PTPM_SYNC,othertime)
now=t2
return t2-t1
delay_packet
send PTPM_DELAY
receive PTPM_DELAY othertime=t3
now=t4
return t4-t3
send PTPM_NEXT
*/
static void ptp_send_packet_in(struct master_data * md, int what, ptp_time_t t_send)
{
unsigned char bufout[4+8+8]; // PTP# timein timeout
memcpy(bufout,"PTP",3);
bufout[3] = what;
memcpy(bufout+4,&t_send,8);
memset(bufout+12,0,8);
ptp_send_packet(md->sock,bufout,sizeof(bufout),md->clientdata,md->clientdatasize);
}
enum MasterState { MASTER_START, MASTER_WAIT0,MASTER_TIMES, MASTER_SYNC, MASTER_SYNCANS, MASTER_DELAY, MASTER_DELAYANS,MASTER_LOOPEND, MASTER_DONE};
static const char * names[] = {"PTPM_START", "PTPM_SYNC", "PTPM_TIMES", "PTPM_DELAY", "PTPM_NEXT", "PPTS_HELLO"};
int master_sm(struct master_data * md, enum Event e, unsigned char * data, int n)
{
int ptype = -1;
ptp_time_t treceived;
ptp_time_t treceived2;
// multiple steps
if(e == EVENT_RESET)
{
md->bigstate = MASTER_START;
}
else if(e == EVENT_NEWPACKET)
{
if(n == 20 && strncmp((char*)data,"PTP",3) == 0)
{
// extract for the state machine and dump
ptype = (int)data[3];
treceived = *(ptp_time_t*)(data+4);
treceived2 = *(ptp_time_t*)(data+12);
printf("master received %s(%d) with t=%lld t2=%lld\n",ptype >= 0 && ptype < PTPX_MAX ? names[ptype] : "unknown",ptype,treceived,treceived2);
}
else
{
printf("wrong packet data: length %d %s\n",n,data);
}
}
while(1)
switch(md->bigstate)
{
case MASTER_START:
md-> largest_offset = LONG_MIN;
md-> smallest_offset = LONG_MAX;
md-> sum_offset = 0;
md-> smallest_delay = LONG_MAX;
md-> largest_delay = LONG_MIN;
md-> sum_delay = 0;
md-> i = 0;
md-> bigstate = MASTER_WAIT0;
ptp_send_packet_in(md,PTPM_START,0);
return 0; // wait
case MASTER_WAIT0:
if(e == EVENT_NEWPACKET)
{
if(ptype != PTPM_START)
{
printf("expected START got %d\n",ptype);
md->bigstate = MASTER_START;
break; // retry
}
else
{
// check PTPM_START otherwise go MASTER_START
ptp_send_packet_in(md,PTPM_TIMES,md->nsteps);
md->bigstate = MASTER_TIMES;
return 0; // wait
}
}
return 0; // ignore
case MASTER_TIMES:
if(e == EVENT_NEWPACKET)
{
if(ptype != PTPM_TIMES)
{
printf("expected TIMES\n");
md->bigstate = MASTER_START;
break; // retry
}
else
{
md->bigstate = MASTER_SYNC;
break; // next state
}
}
return 0; // ignore
case MASTER_SYNC:
{
// IMPORTANT: in real PTP we use the clock at the moment of the NIC exit
// NOTE: under Linux with SO_TIMESTAMPING it is possible to know when the packet left the NIC so we'll use SYNCDETAIL
ptp_time_t t1;
// sync is: get time, send, get answer, measure offset as difference between answer and original
ptp_get_time(&t1);
ptp_send_packet_in(md,PTPM_SYNC,t1);
md->bigstate = MASTER_SYNCANS;
}
return 0; // wait
case MASTER_SYNCANS:
if(e == EVENT_NEWPACKET)
{
if(ptype != PTPM_SYNC)
{
printf("expected SYNC\n");
md->bigstate = MASTER_START;
break; // restart
}
md->ms_diff = (TO_NSEC(treceived2) - TO_NSEC(treceived)); // t2-t1
md->bigstate = MASTER_DELAY;
break; // next state
}
else
return 0; // ignore
case MASTER_DELAY:
{
// this is a REQUEST for DELAY we don't care about master time here
ptp_time_t tmp;
ptp_get_time(&tmp);
ptp_send_packet_in(md,PTPM_DELAY,tmp); // NOT used in protocol
md->bigstate = MASTER_DELAYANS;
}
return 0; // wait
case MASTER_DELAYANS:
if(e == EVENT_NEWPACKET)
{
if(ptype != PTPM_DELAY)
{
printf("expected delay!");
md->bigstate = MASTER_START;
break; // restart
}
else
{
// TODO: replace it with the SOCKET timestamp if SO_TIMESTAMP available
ptp_time_t t4;
ptp_get_time(&t4); // == t4
if(md->alttime != 0)
{
ptp_time_t a = TO_NSEC(md->alttime);
ptp_time_t b = TO_NSEC(t4);
if(a != b)
{
printf("delta %lld\n",b-a);
t4 = md->alttime;
}
else
printf("nodelta\n");
}
md->sm_diff = (TO_NSEC(t4) - TO_NSEC(treceived2)); // t4-t3
md->bigstate = MASTER_LOOPEND;
break; // next state
}
}
return 0; // ignore
case MASTER_LOOPEND:
{
long long offset = (md->ms_diff - md->sm_diff)/2;
long long delay = (md->ms_diff + md->sm_diff)/2;
/* calculate averages, min, max */
md->sum_offset += offset;
if (md->largest_offset < offset) {
md->largest_offset = offset;
}
if (md->smallest_offset > offset) {
md->smallest_offset = offset;
}
md->sum_delay += delay;
if (md->largest_delay < delay) {
md->largest_delay = delay;
}
if (md->smallest_delay > delay) {
md->smallest_delay = delay;
}
if(++md->i >= md->nsteps)
{
md->bigstate = MASTER_DONE;
md->steps = md->i;
return 1; // completion
}
else
{
ptp_send_packet_in(md, PTPM_NEXT, 0);
md->bigstate = MASTER_SYNC;
return 0; // wait
}
}
default:
return 0; // ignore
}
}