• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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 	size_t if_len;
317 
318 	if (argc < 2)
319 		usage(0);
320 	interface = argv[1];
321 	if_len = strlen(interface);
322 	if (if_len >= IFNAMSIZ) {
323 		printf("interface name exceeds IFNAMSIZ\n");
324 		exit(1);
325 	}
326 
327 	for (i = 2; i < argc; i++) {
328 		if (!strcasecmp(argv[i], "SO_TIMESTAMP"))
329 			so_timestamp = 1;
330 		else if (!strcasecmp(argv[i], "SO_TIMESTAMPNS"))
331 			so_timestampns = 1;
332 		else if (!strcasecmp(argv[i], "SIOCGSTAMP"))
333 			siocgstamp = 1;
334 		else if (!strcasecmp(argv[i], "SIOCGSTAMPNS"))
335 			siocgstampns = 1;
336 		else if (!strcasecmp(argv[i], "IP_MULTICAST_LOOP"))
337 			ip_multicast_loop = 1;
338 		else if (!strcasecmp(argv[i], "SOF_TIMESTAMPING_TX_HARDWARE"))
339 			so_timestamping_flags |= SOF_TIMESTAMPING_TX_HARDWARE;
340 		else if (!strcasecmp(argv[i], "SOF_TIMESTAMPING_TX_SOFTWARE"))
341 			so_timestamping_flags |= SOF_TIMESTAMPING_TX_SOFTWARE;
342 		else if (!strcasecmp(argv[i], "SOF_TIMESTAMPING_RX_HARDWARE"))
343 			so_timestamping_flags |= SOF_TIMESTAMPING_RX_HARDWARE;
344 		else if (!strcasecmp(argv[i], "SOF_TIMESTAMPING_RX_SOFTWARE"))
345 			so_timestamping_flags |= SOF_TIMESTAMPING_RX_SOFTWARE;
346 		else if (!strcasecmp(argv[i], "SOF_TIMESTAMPING_SOFTWARE"))
347 			so_timestamping_flags |= SOF_TIMESTAMPING_SOFTWARE;
348 		else if (!strcasecmp(argv[i], "SOF_TIMESTAMPING_RAW_HARDWARE"))
349 			so_timestamping_flags |= SOF_TIMESTAMPING_RAW_HARDWARE;
350 		else
351 			usage(argv[i]);
352 	}
353 
354 	sock = socket(PF_INET, SOCK_DGRAM, IPPROTO_UDP);
355 	if (sock < 0)
356 		bail("socket");
357 
358 	memset(&device, 0, sizeof(device));
359 	memcpy(device.ifr_name, interface, if_len + 1);
360 	if (ioctl(sock, SIOCGIFADDR, &device) < 0)
361 		bail("getting interface IP address");
362 
363 	memset(&hwtstamp, 0, sizeof(hwtstamp));
364 	memcpy(hwtstamp.ifr_name, interface, if_len + 1);
365 	hwtstamp.ifr_data = (void *)&hwconfig;
366 	memset(&hwconfig, 0, sizeof(hwconfig));
367 	hwconfig.tx_type =
368 		(so_timestamping_flags & SOF_TIMESTAMPING_TX_HARDWARE) ?
369 		HWTSTAMP_TX_ON : HWTSTAMP_TX_OFF;
370 	hwconfig.rx_filter =
371 		(so_timestamping_flags & SOF_TIMESTAMPING_RX_HARDWARE) ?
372 		HWTSTAMP_FILTER_PTP_V1_L4_SYNC : HWTSTAMP_FILTER_NONE;
373 	hwconfig_requested = hwconfig;
374 	if (ioctl(sock, SIOCSHWTSTAMP, &hwtstamp) < 0) {
375 		if ((errno == EINVAL || errno == ENOTSUP) &&
376 		    hwconfig_requested.tx_type == HWTSTAMP_TX_OFF &&
377 		    hwconfig_requested.rx_filter == HWTSTAMP_FILTER_NONE)
378 			printf("SIOCSHWTSTAMP: disabling hardware time stamping not possible\n");
379 		else
380 			bail("SIOCSHWTSTAMP");
381 	}
382 	printf("SIOCSHWTSTAMP: tx_type %d requested, got %d; rx_filter %d requested, got %d\n",
383 	       hwconfig_requested.tx_type, hwconfig.tx_type,
384 	       hwconfig_requested.rx_filter, hwconfig.rx_filter);
385 
386 	/* bind to PTP port */
387 	addr.sin_family = AF_INET;
388 	addr.sin_addr.s_addr = htonl(INADDR_ANY);
389 	addr.sin_port = htons(319 /* PTP event port */);
390 	if (bind(sock,
391 		 (struct sockaddr *)&addr,
392 		 sizeof(struct sockaddr_in)) < 0)
393 		bail("bind");
394 
395 	/* set multicast group for outgoing packets */
396 	inet_aton("224.0.1.130", &iaddr); /* alternate PTP domain 1 */
397 	addr.sin_addr = iaddr;
398 	imr.imr_multiaddr.s_addr = iaddr.s_addr;
399 	imr.imr_interface.s_addr =
400 		((struct sockaddr_in *)&device.ifr_addr)->sin_addr.s_addr;
401 	if (setsockopt(sock, IPPROTO_IP, IP_MULTICAST_IF,
402 		       &imr.imr_interface.s_addr, sizeof(struct in_addr)) < 0)
403 		bail("set multicast");
404 
405 	/* join multicast group, loop our own packet */
406 	if (setsockopt(sock, IPPROTO_IP, IP_ADD_MEMBERSHIP,
407 		       &imr, sizeof(struct ip_mreq)) < 0)
408 		bail("join multicast group");
409 
410 	if (setsockopt(sock, IPPROTO_IP, IP_MULTICAST_LOOP,
411 		       &ip_multicast_loop, sizeof(enabled)) < 0) {
412 		bail("loop multicast");
413 	}
414 
415 	/* set socket options for time stamping */
416 	if (so_timestamp &&
417 		setsockopt(sock, SOL_SOCKET, SO_TIMESTAMP,
418 			   &enabled, sizeof(enabled)) < 0)
419 		bail("setsockopt SO_TIMESTAMP");
420 
421 	if (so_timestampns &&
422 		setsockopt(sock, SOL_SOCKET, SO_TIMESTAMPNS,
423 			   &enabled, sizeof(enabled)) < 0)
424 		bail("setsockopt SO_TIMESTAMPNS");
425 
426 	if (so_timestamping_flags &&
427 		setsockopt(sock, SOL_SOCKET, SO_TIMESTAMPING,
428 			   &so_timestamping_flags,
429 			   sizeof(so_timestamping_flags)) < 0)
430 		bail("setsockopt SO_TIMESTAMPING");
431 
432 	/* request IP_PKTINFO for debugging purposes */
433 	if (setsockopt(sock, SOL_IP, IP_PKTINFO,
434 		       &enabled, sizeof(enabled)) < 0)
435 		printf("%s: %s\n", "setsockopt IP_PKTINFO", strerror(errno));
436 
437 	/* verify socket options */
438 	len = sizeof(val);
439 	if (getsockopt(sock, SOL_SOCKET, SO_TIMESTAMP, &val, &len) < 0)
440 		printf("%s: %s\n", "getsockopt SO_TIMESTAMP", strerror(errno));
441 	else
442 		printf("SO_TIMESTAMP %d\n", val);
443 
444 	if (getsockopt(sock, SOL_SOCKET, SO_TIMESTAMPNS, &val, &len) < 0)
445 		printf("%s: %s\n", "getsockopt SO_TIMESTAMPNS",
446 		       strerror(errno));
447 	else
448 		printf("SO_TIMESTAMPNS %d\n", val);
449 
450 	if (getsockopt(sock, SOL_SOCKET, SO_TIMESTAMPING, &val, &len) < 0) {
451 		printf("%s: %s\n", "getsockopt SO_TIMESTAMPING",
452 		       strerror(errno));
453 	} else {
454 		printf("SO_TIMESTAMPING %d\n", val);
455 		if (val != so_timestamping_flags)
456 			printf("   not the expected value %d\n",
457 			       so_timestamping_flags);
458 	}
459 
460 	/* send packets forever every five seconds */
461 	gettimeofday(&next, 0);
462 	next.tv_sec = (next.tv_sec + 1) / 5 * 5;
463 	next.tv_usec = 0;
464 	while (1) {
465 		struct timeval now;
466 		struct timeval delta;
467 		long delta_us;
468 		int res;
469 		fd_set readfs, errorfs;
470 
471 		gettimeofday(&now, 0);
472 		delta_us = (long)(next.tv_sec - now.tv_sec) * 1000000 +
473 			(long)(next.tv_usec - now.tv_usec);
474 		if (delta_us > 0) {
475 			/* continue waiting for timeout or data */
476 			delta.tv_sec = delta_us / 1000000;
477 			delta.tv_usec = delta_us % 1000000;
478 
479 			FD_ZERO(&readfs);
480 			FD_ZERO(&errorfs);
481 			FD_SET(sock, &readfs);
482 			FD_SET(sock, &errorfs);
483 			printf("%ld.%06ld: select %ldus\n",
484 			       (long)now.tv_sec, (long)now.tv_usec,
485 			       delta_us);
486 			res = select(sock + 1, &readfs, 0, &errorfs, &delta);
487 			gettimeofday(&now, 0);
488 			printf("%ld.%06ld: select returned: %d, %s\n",
489 			       (long)now.tv_sec, (long)now.tv_usec,
490 			       res,
491 			       res < 0 ? strerror(errno) : "success");
492 			if (res > 0) {
493 				if (FD_ISSET(sock, &readfs))
494 					printf("ready for reading\n");
495 				if (FD_ISSET(sock, &errorfs))
496 					printf("has error\n");
497 				recvpacket(sock, 0,
498 					   siocgstamp,
499 					   siocgstampns);
500 				recvpacket(sock, MSG_ERRQUEUE,
501 					   siocgstamp,
502 					   siocgstampns);
503 			}
504 		} else {
505 			/* write one packet */
506 			sendpacket(sock,
507 				   (struct sockaddr *)&addr,
508 				   sizeof(addr));
509 			next.tv_sec += 5;
510 			continue;
511 		}
512 	}
513 
514 	return 0;
515 }
516