1 // SPDX-License-Identifier: GPL-2.0-only
2 /*
3 * This program demonstrates how the various time stamping features in
4 * the Linux kernel work. It emulates the behavior of a PTP
5 * implementation in stand-alone master mode by sending PTPv1 Sync
6 * multicasts once every second. It looks for similar packets, but
7 * beyond that doesn't actually implement PTP.
8 *
9 * Outgoing packets are time stamped with SO_TIMESTAMPING with or
10 * without hardware support.
11 *
12 * Incoming packets are time stamped with SO_TIMESTAMPING with or
13 * without hardware support, SIOCGSTAMP[NS] (per-socket time stamp) and
14 * SO_TIMESTAMP[NS].
15 *
16 * Copyright (C) 2009 Intel Corporation.
17 * Author: Patrick Ohly <patrick.ohly@intel.com>
18 */
19
20 #include <stdio.h>
21 #include <stdlib.h>
22 #include <errno.h>
23 #include <string.h>
24
25 #include <sys/time.h>
26 #include <sys/socket.h>
27 #include <sys/select.h>
28 #include <sys/ioctl.h>
29 #include <arpa/inet.h>
30 #include <net/if.h>
31
32 #include <asm/types.h>
33 #include <linux/net_tstamp.h>
34 #include <linux/errqueue.h>
35 #include <linux/sockios.h>
36
37 #ifndef SO_TIMESTAMPING
38 # define SO_TIMESTAMPING 37
39 # define SCM_TIMESTAMPING SO_TIMESTAMPING
40 #endif
41
42 #ifndef SO_TIMESTAMPNS
43 # define SO_TIMESTAMPNS 35
44 #endif
45
usage(const char * error)46 static void usage(const char *error)
47 {
48 if (error)
49 printf("invalid option: %s\n", error);
50 printf("timestamping interface option*\n\n"
51 "Options:\n"
52 " IP_MULTICAST_LOOP - looping outgoing multicasts\n"
53 " SO_TIMESTAMP - normal software time stamping, ms resolution\n"
54 " SO_TIMESTAMPNS - more accurate software time stamping\n"
55 " SOF_TIMESTAMPING_TX_HARDWARE - hardware time stamping of outgoing packets\n"
56 " SOF_TIMESTAMPING_TX_SOFTWARE - software fallback for outgoing packets\n"
57 " SOF_TIMESTAMPING_RX_HARDWARE - hardware time stamping of incoming packets\n"
58 " SOF_TIMESTAMPING_RX_SOFTWARE - software fallback for incoming packets\n"
59 " SOF_TIMESTAMPING_SOFTWARE - request reporting of software time stamps\n"
60 " SOF_TIMESTAMPING_RAW_HARDWARE - request reporting of raw HW time stamps\n"
61 " SIOCGSTAMP - check last socket time stamp\n"
62 " SIOCGSTAMPNS - more accurate socket time stamp\n");
63 exit(1);
64 }
65
bail(const char * error)66 static void bail(const char *error)
67 {
68 printf("%s: %s\n", error, strerror(errno));
69 exit(1);
70 }
71
72 static const unsigned char sync[] = {
73 0x00, 0x01, 0x00, 0x01,
74 0x5f, 0x44, 0x46, 0x4c,
75 0x54, 0x00, 0x00, 0x00,
76 0x00, 0x00, 0x00, 0x00,
77 0x00, 0x00, 0x00, 0x00,
78 0x01, 0x01,
79
80 /* fake uuid */
81 0x00, 0x01,
82 0x02, 0x03, 0x04, 0x05,
83
84 0x00, 0x01, 0x00, 0x37,
85 0x00, 0x00, 0x00, 0x08,
86 0x00, 0x00, 0x00, 0x00,
87 0x49, 0x05, 0xcd, 0x01,
88 0x29, 0xb1, 0x8d, 0xb0,
89 0x00, 0x00, 0x00, 0x00,
90 0x00, 0x01,
91
92 /* fake uuid */
93 0x00, 0x01,
94 0x02, 0x03, 0x04, 0x05,
95
96 0x00, 0x00, 0x00, 0x37,
97 0x00, 0x00, 0x00, 0x04,
98 0x44, 0x46, 0x4c, 0x54,
99 0x00, 0x00, 0xf0, 0x60,
100 0x00, 0x01, 0x00, 0x00,
101 0x00, 0x00, 0x00, 0x01,
102 0x00, 0x00, 0xf0, 0x60,
103 0x00, 0x00, 0x00, 0x00,
104 0x00, 0x00, 0x00, 0x04,
105 0x44, 0x46, 0x4c, 0x54,
106 0x00, 0x01,
107
108 /* fake uuid */
109 0x00, 0x01,
110 0x02, 0x03, 0x04, 0x05,
111
112 0x00, 0x00, 0x00, 0x00,
113 0x00, 0x00, 0x00, 0x00,
114 0x00, 0x00, 0x00, 0x00,
115 0x00, 0x00, 0x00, 0x00
116 };
117
sendpacket(int sock,struct sockaddr * addr,socklen_t addr_len)118 static void sendpacket(int sock, struct sockaddr *addr, socklen_t addr_len)
119 {
120 struct timeval now;
121 int res;
122
123 res = sendto(sock, sync, sizeof(sync), 0,
124 addr, addr_len);
125 gettimeofday(&now, 0);
126 if (res < 0)
127 printf("%s: %s\n", "send", strerror(errno));
128 else
129 printf("%ld.%06ld: sent %d bytes\n",
130 (long)now.tv_sec, (long)now.tv_usec,
131 res);
132 }
133
printpacket(struct msghdr * msg,int res,char * data,int sock,int recvmsg_flags,int siocgstamp,int siocgstampns)134 static void printpacket(struct msghdr *msg, int res,
135 char *data,
136 int sock, int recvmsg_flags,
137 int siocgstamp, int siocgstampns)
138 {
139 struct sockaddr_in *from_addr = (struct sockaddr_in *)msg->msg_name;
140 struct cmsghdr *cmsg;
141 struct timeval tv;
142 struct timespec ts;
143 struct timeval now;
144
145 gettimeofday(&now, 0);
146
147 printf("%ld.%06ld: received %s data, %d bytes from %s, %zu bytes control messages\n",
148 (long)now.tv_sec, (long)now.tv_usec,
149 (recvmsg_flags & MSG_ERRQUEUE) ? "error" : "regular",
150 res,
151 inet_ntoa(from_addr->sin_addr),
152 msg->msg_controllen);
153 for (cmsg = CMSG_FIRSTHDR(msg);
154 cmsg;
155 cmsg = CMSG_NXTHDR(msg, cmsg)) {
156 printf(" cmsg len %zu: ", cmsg->cmsg_len);
157 switch (cmsg->cmsg_level) {
158 case SOL_SOCKET:
159 printf("SOL_SOCKET ");
160 switch (cmsg->cmsg_type) {
161 case SO_TIMESTAMP: {
162 struct timeval *stamp =
163 (struct timeval *)CMSG_DATA(cmsg);
164 printf("SO_TIMESTAMP %ld.%06ld",
165 (long)stamp->tv_sec,
166 (long)stamp->tv_usec);
167 break;
168 }
169 case SO_TIMESTAMPNS: {
170 struct timespec *stamp =
171 (struct timespec *)CMSG_DATA(cmsg);
172 printf("SO_TIMESTAMPNS %ld.%09ld",
173 (long)stamp->tv_sec,
174 (long)stamp->tv_nsec);
175 break;
176 }
177 case SO_TIMESTAMPING: {
178 struct timespec *stamp =
179 (struct timespec *)CMSG_DATA(cmsg);
180 printf("SO_TIMESTAMPING ");
181 printf("SW %ld.%09ld ",
182 (long)stamp->tv_sec,
183 (long)stamp->tv_nsec);
184 stamp++;
185 /* skip deprecated HW transformed */
186 stamp++;
187 printf("HW raw %ld.%09ld",
188 (long)stamp->tv_sec,
189 (long)stamp->tv_nsec);
190 break;
191 }
192 default:
193 printf("type %d", cmsg->cmsg_type);
194 break;
195 }
196 break;
197 case IPPROTO_IP:
198 printf("IPPROTO_IP ");
199 switch (cmsg->cmsg_type) {
200 case IP_RECVERR: {
201 struct sock_extended_err *err =
202 (struct sock_extended_err *)CMSG_DATA(cmsg);
203 printf("IP_RECVERR ee_errno '%s' ee_origin %d => %s",
204 strerror(err->ee_errno),
205 err->ee_origin,
206 #ifdef SO_EE_ORIGIN_TIMESTAMPING
207 err->ee_origin == SO_EE_ORIGIN_TIMESTAMPING ?
208 "bounced packet" : "unexpected origin"
209 #else
210 "probably SO_EE_ORIGIN_TIMESTAMPING"
211 #endif
212 );
213 if (res < sizeof(sync))
214 printf(" => truncated data?!");
215 else if (!memcmp(sync, data + res - sizeof(sync),
216 sizeof(sync)))
217 printf(" => GOT OUR DATA BACK (HURRAY!)");
218 break;
219 }
220 case IP_PKTINFO: {
221 struct in_pktinfo *pktinfo =
222 (struct in_pktinfo *)CMSG_DATA(cmsg);
223 printf("IP_PKTINFO interface index %u",
224 pktinfo->ipi_ifindex);
225 break;
226 }
227 default:
228 printf("type %d", cmsg->cmsg_type);
229 break;
230 }
231 break;
232 default:
233 printf("level %d type %d",
234 cmsg->cmsg_level,
235 cmsg->cmsg_type);
236 break;
237 }
238 printf("\n");
239 }
240
241 if (siocgstamp) {
242 if (ioctl(sock, SIOCGSTAMP, &tv))
243 printf(" %s: %s\n", "SIOCGSTAMP", strerror(errno));
244 else
245 printf("SIOCGSTAMP %ld.%06ld\n",
246 (long)tv.tv_sec,
247 (long)tv.tv_usec);
248 }
249 if (siocgstampns) {
250 if (ioctl(sock, SIOCGSTAMPNS, &ts))
251 printf(" %s: %s\n", "SIOCGSTAMPNS", strerror(errno));
252 else
253 printf("SIOCGSTAMPNS %ld.%09ld\n",
254 (long)ts.tv_sec,
255 (long)ts.tv_nsec);
256 }
257 }
258
recvpacket(int sock,int recvmsg_flags,int siocgstamp,int siocgstampns)259 static void recvpacket(int sock, int recvmsg_flags,
260 int siocgstamp, int siocgstampns)
261 {
262 char data[256];
263 struct msghdr msg;
264 struct iovec entry;
265 struct sockaddr_in from_addr;
266 struct {
267 struct cmsghdr cm;
268 char control[512];
269 } control;
270 int res;
271
272 memset(&msg, 0, sizeof(msg));
273 msg.msg_iov = &entry;
274 msg.msg_iovlen = 1;
275 entry.iov_base = data;
276 entry.iov_len = sizeof(data);
277 msg.msg_name = (caddr_t)&from_addr;
278 msg.msg_namelen = sizeof(from_addr);
279 msg.msg_control = &control;
280 msg.msg_controllen = sizeof(control);
281
282 res = recvmsg(sock, &msg, recvmsg_flags|MSG_DONTWAIT);
283 if (res < 0) {
284 printf("%s %s: %s\n",
285 "recvmsg",
286 (recvmsg_flags & MSG_ERRQUEUE) ? "error" : "regular",
287 strerror(errno));
288 } else {
289 printpacket(&msg, res, data,
290 sock, recvmsg_flags,
291 siocgstamp, siocgstampns);
292 }
293 }
294
main(int argc,char ** argv)295 int main(int argc, char **argv)
296 {
297 int so_timestamping_flags = 0;
298 int so_timestamp = 0;
299 int so_timestampns = 0;
300 int siocgstamp = 0;
301 int siocgstampns = 0;
302 int ip_multicast_loop = 0;
303 char *interface;
304 int i;
305 int enabled = 1;
306 int sock;
307 struct ifreq device;
308 struct ifreq hwtstamp;
309 struct hwtstamp_config hwconfig, hwconfig_requested;
310 struct sockaddr_in addr;
311 struct ip_mreq imr;
312 struct in_addr iaddr;
313 int val;
314 socklen_t len;
315 struct timeval next;
316
317 if (argc < 2)
318 usage(0);
319 interface = argv[1];
320
321 for (i = 2; i < argc; i++) {
322 if (!strcasecmp(argv[i], "SO_TIMESTAMP"))
323 so_timestamp = 1;
324 else if (!strcasecmp(argv[i], "SO_TIMESTAMPNS"))
325 so_timestampns = 1;
326 else if (!strcasecmp(argv[i], "SIOCGSTAMP"))
327 siocgstamp = 1;
328 else if (!strcasecmp(argv[i], "SIOCGSTAMPNS"))
329 siocgstampns = 1;
330 else if (!strcasecmp(argv[i], "IP_MULTICAST_LOOP"))
331 ip_multicast_loop = 1;
332 else if (!strcasecmp(argv[i], "SOF_TIMESTAMPING_TX_HARDWARE"))
333 so_timestamping_flags |= SOF_TIMESTAMPING_TX_HARDWARE;
334 else if (!strcasecmp(argv[i], "SOF_TIMESTAMPING_TX_SOFTWARE"))
335 so_timestamping_flags |= SOF_TIMESTAMPING_TX_SOFTWARE;
336 else if (!strcasecmp(argv[i], "SOF_TIMESTAMPING_RX_HARDWARE"))
337 so_timestamping_flags |= SOF_TIMESTAMPING_RX_HARDWARE;
338 else if (!strcasecmp(argv[i], "SOF_TIMESTAMPING_RX_SOFTWARE"))
339 so_timestamping_flags |= SOF_TIMESTAMPING_RX_SOFTWARE;
340 else if (!strcasecmp(argv[i], "SOF_TIMESTAMPING_SOFTWARE"))
341 so_timestamping_flags |= SOF_TIMESTAMPING_SOFTWARE;
342 else if (!strcasecmp(argv[i], "SOF_TIMESTAMPING_RAW_HARDWARE"))
343 so_timestamping_flags |= SOF_TIMESTAMPING_RAW_HARDWARE;
344 else
345 usage(argv[i]);
346 }
347
348 sock = socket(PF_INET, SOCK_DGRAM, IPPROTO_UDP);
349 if (sock < 0)
350 bail("socket");
351
352 memset(&device, 0, sizeof(device));
353 strncpy(device.ifr_name, interface, sizeof(device.ifr_name));
354 if (ioctl(sock, SIOCGIFADDR, &device) < 0)
355 bail("getting interface IP address");
356
357 memset(&hwtstamp, 0, sizeof(hwtstamp));
358 strncpy(hwtstamp.ifr_name, interface, sizeof(hwtstamp.ifr_name));
359 hwtstamp.ifr_data = (void *)&hwconfig;
360 memset(&hwconfig, 0, sizeof(hwconfig));
361 hwconfig.tx_type =
362 (so_timestamping_flags & SOF_TIMESTAMPING_TX_HARDWARE) ?
363 HWTSTAMP_TX_ON : HWTSTAMP_TX_OFF;
364 hwconfig.rx_filter =
365 (so_timestamping_flags & SOF_TIMESTAMPING_RX_HARDWARE) ?
366 HWTSTAMP_FILTER_PTP_V1_L4_SYNC : HWTSTAMP_FILTER_NONE;
367 hwconfig_requested = hwconfig;
368 if (ioctl(sock, SIOCSHWTSTAMP, &hwtstamp) < 0) {
369 if ((errno == EINVAL || errno == ENOTSUP) &&
370 hwconfig_requested.tx_type == HWTSTAMP_TX_OFF &&
371 hwconfig_requested.rx_filter == HWTSTAMP_FILTER_NONE)
372 printf("SIOCSHWTSTAMP: disabling hardware time stamping not possible\n");
373 else
374 bail("SIOCSHWTSTAMP");
375 }
376 printf("SIOCSHWTSTAMP: tx_type %d requested, got %d; rx_filter %d requested, got %d\n",
377 hwconfig_requested.tx_type, hwconfig.tx_type,
378 hwconfig_requested.rx_filter, hwconfig.rx_filter);
379
380 /* bind to PTP port */
381 addr.sin_family = AF_INET;
382 addr.sin_addr.s_addr = htonl(INADDR_ANY);
383 addr.sin_port = htons(319 /* PTP event port */);
384 if (bind(sock,
385 (struct sockaddr *)&addr,
386 sizeof(struct sockaddr_in)) < 0)
387 bail("bind");
388
389 /* set multicast group for outgoing packets */
390 inet_aton("224.0.1.130", &iaddr); /* alternate PTP domain 1 */
391 addr.sin_addr = iaddr;
392 imr.imr_multiaddr.s_addr = iaddr.s_addr;
393 imr.imr_interface.s_addr =
394 ((struct sockaddr_in *)&device.ifr_addr)->sin_addr.s_addr;
395 if (setsockopt(sock, IPPROTO_IP, IP_MULTICAST_IF,
396 &imr.imr_interface.s_addr, sizeof(struct in_addr)) < 0)
397 bail("set multicast");
398
399 /* join multicast group, loop our own packet */
400 if (setsockopt(sock, IPPROTO_IP, IP_ADD_MEMBERSHIP,
401 &imr, sizeof(struct ip_mreq)) < 0)
402 bail("join multicast group");
403
404 if (setsockopt(sock, IPPROTO_IP, IP_MULTICAST_LOOP,
405 &ip_multicast_loop, sizeof(enabled)) < 0) {
406 bail("loop multicast");
407 }
408
409 /* set socket options for time stamping */
410 if (so_timestamp &&
411 setsockopt(sock, SOL_SOCKET, SO_TIMESTAMP,
412 &enabled, sizeof(enabled)) < 0)
413 bail("setsockopt SO_TIMESTAMP");
414
415 if (so_timestampns &&
416 setsockopt(sock, SOL_SOCKET, SO_TIMESTAMPNS,
417 &enabled, sizeof(enabled)) < 0)
418 bail("setsockopt SO_TIMESTAMPNS");
419
420 if (so_timestamping_flags &&
421 setsockopt(sock, SOL_SOCKET, SO_TIMESTAMPING,
422 &so_timestamping_flags,
423 sizeof(so_timestamping_flags)) < 0)
424 bail("setsockopt SO_TIMESTAMPING");
425
426 /* request IP_PKTINFO for debugging purposes */
427 if (setsockopt(sock, SOL_IP, IP_PKTINFO,
428 &enabled, sizeof(enabled)) < 0)
429 printf("%s: %s\n", "setsockopt IP_PKTINFO", strerror(errno));
430
431 /* verify socket options */
432 len = sizeof(val);
433 if (getsockopt(sock, SOL_SOCKET, SO_TIMESTAMP, &val, &len) < 0)
434 printf("%s: %s\n", "getsockopt SO_TIMESTAMP", strerror(errno));
435 else
436 printf("SO_TIMESTAMP %d\n", val);
437
438 if (getsockopt(sock, SOL_SOCKET, SO_TIMESTAMPNS, &val, &len) < 0)
439 printf("%s: %s\n", "getsockopt SO_TIMESTAMPNS",
440 strerror(errno));
441 else
442 printf("SO_TIMESTAMPNS %d\n", val);
443
444 if (getsockopt(sock, SOL_SOCKET, SO_TIMESTAMPING, &val, &len) < 0) {
445 printf("%s: %s\n", "getsockopt SO_TIMESTAMPING",
446 strerror(errno));
447 } else {
448 printf("SO_TIMESTAMPING %d\n", val);
449 if (val != so_timestamping_flags)
450 printf(" not the expected value %d\n",
451 so_timestamping_flags);
452 }
453
454 /* send packets forever every five seconds */
455 gettimeofday(&next, 0);
456 next.tv_sec = (next.tv_sec + 1) / 5 * 5;
457 next.tv_usec = 0;
458 while (1) {
459 struct timeval now;
460 struct timeval delta;
461 long delta_us;
462 int res;
463 fd_set readfs, errorfs;
464
465 gettimeofday(&now, 0);
466 delta_us = (long)(next.tv_sec - now.tv_sec) * 1000000 +
467 (long)(next.tv_usec - now.tv_usec);
468 if (delta_us > 0) {
469 /* continue waiting for timeout or data */
470 delta.tv_sec = delta_us / 1000000;
471 delta.tv_usec = delta_us % 1000000;
472
473 FD_ZERO(&readfs);
474 FD_ZERO(&errorfs);
475 FD_SET(sock, &readfs);
476 FD_SET(sock, &errorfs);
477 printf("%ld.%06ld: select %ldus\n",
478 (long)now.tv_sec, (long)now.tv_usec,
479 delta_us);
480 res = select(sock + 1, &readfs, 0, &errorfs, &delta);
481 gettimeofday(&now, 0);
482 printf("%ld.%06ld: select returned: %d, %s\n",
483 (long)now.tv_sec, (long)now.tv_usec,
484 res,
485 res < 0 ? strerror(errno) : "success");
486 if (res > 0) {
487 if (FD_ISSET(sock, &readfs))
488 printf("ready for reading\n");
489 if (FD_ISSET(sock, &errorfs))
490 printf("has error\n");
491 recvpacket(sock, 0,
492 siocgstamp,
493 siocgstampns);
494 recvpacket(sock, MSG_ERRQUEUE,
495 siocgstamp,
496 siocgstampns);
497 }
498 } else {
499 /* write one packet */
500 sendpacket(sock,
501 (struct sockaddr *)&addr,
502 sizeof(addr));
503 next.tv_sec += 5;
504 continue;
505 }
506 }
507
508 return 0;
509 }
510