txtimestamp.c 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /*
  3. * Copyright 2014 Google Inc.
  4. * Author: willemb@google.com (Willem de Bruijn)
  5. *
  6. * Test software tx timestamping, including
  7. *
  8. * - SCHED, SND and ACK timestamps
  9. * - RAW, UDP and TCP
  10. * - IPv4 and IPv6
  11. * - various packet sizes (to test GSO and TSO)
  12. *
  13. * Consult the command line arguments for help on running
  14. * the various testcases.
  15. *
  16. * This test requires a dummy TCP server.
  17. * A simple `nc6 [-u] -l -p $DESTPORT` will do
  18. */
  19. #define _GNU_SOURCE
  20. #include <arpa/inet.h>
  21. #include <asm/types.h>
  22. #include <error.h>
  23. #include <errno.h>
  24. #include <inttypes.h>
  25. #include <linux/errqueue.h>
  26. #include <linux/if_ether.h>
  27. #include <linux/if_packet.h>
  28. #include <linux/ipv6.h>
  29. #include <linux/net_tstamp.h>
  30. #include <netdb.h>
  31. #include <net/if.h>
  32. #include <netinet/in.h>
  33. #include <netinet/ip.h>
  34. #include <netinet/udp.h>
  35. #include <netinet/tcp.h>
  36. #include <poll.h>
  37. #include <stdarg.h>
  38. #include <stdbool.h>
  39. #include <stdio.h>
  40. #include <stdlib.h>
  41. #include <string.h>
  42. #include <sys/epoll.h>
  43. #include <sys/ioctl.h>
  44. #include <sys/select.h>
  45. #include <sys/socket.h>
  46. #include <sys/time.h>
  47. #include <sys/types.h>
  48. #include <time.h>
  49. #include <unistd.h>
  50. #define NSEC_PER_USEC 1000L
  51. #define USEC_PER_SEC 1000000L
  52. #define NSEC_PER_SEC 1000000000LL
  53. /* command line parameters */
  54. static int cfg_proto = SOCK_STREAM;
  55. static int cfg_ipproto = IPPROTO_TCP;
  56. static int cfg_num_pkts = 4;
  57. static int do_ipv4 = 1;
  58. static int do_ipv6 = 1;
  59. static int cfg_payload_len = 10;
  60. static int cfg_poll_timeout = 100;
  61. static int cfg_delay_snd;
  62. static int cfg_delay_ack;
  63. static int cfg_delay_tolerance_usec = 500;
  64. static bool cfg_show_payload;
  65. static bool cfg_do_pktinfo;
  66. static bool cfg_busy_poll;
  67. static int cfg_sleep_usec = 50 * 1000;
  68. static bool cfg_loop_nodata;
  69. static bool cfg_use_cmsg;
  70. static bool cfg_use_pf_packet;
  71. static bool cfg_use_epoll;
  72. static bool cfg_epollet;
  73. static bool cfg_do_listen;
  74. static uint16_t dest_port = 9000;
  75. static bool cfg_print_nsec;
  76. static uint32_t ts_opt_id;
  77. static bool cfg_use_cmsg_opt_id;
  78. static struct sockaddr_in daddr;
  79. static struct sockaddr_in6 daddr6;
  80. static struct timespec ts_usr;
  81. static int saved_tskey = -1;
  82. static int saved_tskey_type = -1;
  83. struct timing_event {
  84. int64_t min;
  85. int64_t max;
  86. int64_t total;
  87. int count;
  88. };
  89. static struct timing_event usr_enq;
  90. static struct timing_event usr_snd;
  91. static struct timing_event usr_ack;
  92. static bool test_failed;
  93. static int64_t timespec_to_ns64(struct timespec *ts)
  94. {
  95. return ts->tv_sec * NSEC_PER_SEC + ts->tv_nsec;
  96. }
  97. static int64_t timespec_to_us64(struct timespec *ts)
  98. {
  99. return ts->tv_sec * USEC_PER_SEC + ts->tv_nsec / NSEC_PER_USEC;
  100. }
  101. static void init_timing_event(struct timing_event *te)
  102. {
  103. te->min = INT64_MAX;
  104. te->max = 0;
  105. te->total = 0;
  106. te->count = 0;
  107. }
  108. static void add_timing_event(struct timing_event *te,
  109. struct timespec *t_start, struct timespec *t_end)
  110. {
  111. int64_t ts_delta = timespec_to_ns64(t_end) - timespec_to_ns64(t_start);
  112. te->count++;
  113. if (ts_delta < te->min)
  114. te->min = ts_delta;
  115. if (ts_delta > te->max)
  116. te->max = ts_delta;
  117. te->total += ts_delta;
  118. }
  119. static void validate_key(int tskey, int tstype)
  120. {
  121. int stepsize;
  122. /* compare key for each subsequent request
  123. * must only test for one type, the first one requested
  124. */
  125. if (saved_tskey == -1 || cfg_use_cmsg_opt_id)
  126. saved_tskey_type = tstype;
  127. else if (saved_tskey_type != tstype)
  128. return;
  129. stepsize = cfg_proto == SOCK_STREAM ? cfg_payload_len : 1;
  130. stepsize = cfg_use_cmsg_opt_id ? 0 : stepsize;
  131. if (tskey != saved_tskey + stepsize) {
  132. fprintf(stderr, "ERROR: key %d, expected %d\n",
  133. tskey, saved_tskey + stepsize);
  134. test_failed = true;
  135. }
  136. saved_tskey = tskey;
  137. }
  138. static void validate_timestamp(struct timespec *cur, int min_delay)
  139. {
  140. int64_t cur64, start64;
  141. int max_delay;
  142. cur64 = timespec_to_us64(cur);
  143. start64 = timespec_to_us64(&ts_usr);
  144. max_delay = min_delay + cfg_delay_tolerance_usec;
  145. if (cur64 < start64 + min_delay || cur64 > start64 + max_delay) {
  146. fprintf(stderr, "ERROR: %" PRId64 " us expected between %d and %d\n",
  147. cur64 - start64, min_delay, max_delay);
  148. if (!getenv("KSFT_MACHINE_SLOW"))
  149. test_failed = true;
  150. }
  151. }
  152. static void __print_ts_delta_formatted(int64_t ts_delta)
  153. {
  154. if (cfg_print_nsec)
  155. fprintf(stderr, "%" PRId64 " ns", ts_delta);
  156. else
  157. fprintf(stderr, "%" PRId64 " us", ts_delta / NSEC_PER_USEC);
  158. }
  159. static void __print_timestamp(const char *name, struct timespec *cur,
  160. uint32_t key, int payload_len)
  161. {
  162. int64_t ts_delta;
  163. if (!(cur->tv_sec | cur->tv_nsec))
  164. return;
  165. if (cfg_print_nsec)
  166. fprintf(stderr, " %s: %lu s %lu ns (seq=%u, len=%u)",
  167. name, cur->tv_sec, cur->tv_nsec,
  168. key, payload_len);
  169. else
  170. fprintf(stderr, " %s: %lu s %lu us (seq=%u, len=%u)",
  171. name, cur->tv_sec, cur->tv_nsec / NSEC_PER_USEC,
  172. key, payload_len);
  173. if (cur != &ts_usr) {
  174. ts_delta = timespec_to_ns64(cur) - timespec_to_ns64(&ts_usr);
  175. fprintf(stderr, " (USR +");
  176. __print_ts_delta_formatted(ts_delta);
  177. fprintf(stderr, ")");
  178. }
  179. fprintf(stderr, "\n");
  180. }
  181. static void record_timestamp_usr(void)
  182. {
  183. if (clock_gettime(CLOCK_REALTIME, &ts_usr))
  184. error(1, errno, "clock_gettime");
  185. }
  186. static void print_timestamp(struct scm_timestamping *tss, int tstype,
  187. int tskey, int payload_len)
  188. {
  189. const char *tsname = NULL;
  190. validate_key(tskey, tstype);
  191. switch (tstype) {
  192. case SCM_TSTAMP_SCHED:
  193. tsname = " ENQ";
  194. validate_timestamp(&tss->ts[0], 0);
  195. add_timing_event(&usr_enq, &ts_usr, &tss->ts[0]);
  196. break;
  197. case SCM_TSTAMP_SND:
  198. tsname = " SND";
  199. validate_timestamp(&tss->ts[0], cfg_delay_snd);
  200. add_timing_event(&usr_snd, &ts_usr, &tss->ts[0]);
  201. break;
  202. case SCM_TSTAMP_ACK:
  203. tsname = " ACK";
  204. validate_timestamp(&tss->ts[0], cfg_delay_ack);
  205. add_timing_event(&usr_ack, &ts_usr, &tss->ts[0]);
  206. break;
  207. default:
  208. error(1, 0, "unknown timestamp type: %u",
  209. tstype);
  210. }
  211. __print_timestamp(tsname, &tss->ts[0], tskey, payload_len);
  212. }
  213. static void print_timing_event(char *name, struct timing_event *te)
  214. {
  215. if (!te->count)
  216. return;
  217. fprintf(stderr, " %s: count=%d", name, te->count);
  218. fprintf(stderr, ", avg=");
  219. __print_ts_delta_formatted((int64_t)(te->total / te->count));
  220. fprintf(stderr, ", min=");
  221. __print_ts_delta_formatted(te->min);
  222. fprintf(stderr, ", max=");
  223. __print_ts_delta_formatted(te->max);
  224. fprintf(stderr, "\n");
  225. }
  226. /* TODO: convert to check_and_print payload once API is stable */
  227. static void print_payload(char *data, int len)
  228. {
  229. int i;
  230. if (!len)
  231. return;
  232. if (len > 70)
  233. len = 70;
  234. fprintf(stderr, "payload: ");
  235. for (i = 0; i < len; i++)
  236. fprintf(stderr, "%02hhx ", data[i]);
  237. fprintf(stderr, "\n");
  238. }
  239. static void print_pktinfo(int family, int ifindex, void *saddr, void *daddr)
  240. {
  241. char sa[INET6_ADDRSTRLEN], da[INET6_ADDRSTRLEN];
  242. fprintf(stderr, " pktinfo: ifindex=%u src=%s dst=%s\n",
  243. ifindex,
  244. saddr ? inet_ntop(family, saddr, sa, sizeof(sa)) : "unknown",
  245. daddr ? inet_ntop(family, daddr, da, sizeof(da)) : "unknown");
  246. }
  247. static void __epoll(int epfd)
  248. {
  249. struct epoll_event events;
  250. int ret;
  251. memset(&events, 0, sizeof(events));
  252. ret = epoll_wait(epfd, &events, 1, cfg_poll_timeout);
  253. if (ret != 1)
  254. error(1, errno, "epoll_wait");
  255. }
  256. static void __poll(int fd)
  257. {
  258. struct pollfd pollfd;
  259. int ret;
  260. memset(&pollfd, 0, sizeof(pollfd));
  261. pollfd.fd = fd;
  262. ret = poll(&pollfd, 1, cfg_poll_timeout);
  263. if (ret != 1)
  264. error(1, errno, "poll");
  265. }
  266. static void __recv_errmsg_cmsg(struct msghdr *msg, int payload_len)
  267. {
  268. struct sock_extended_err *serr = NULL;
  269. struct scm_timestamping *tss = NULL;
  270. struct cmsghdr *cm;
  271. int batch = 0;
  272. for (cm = CMSG_FIRSTHDR(msg);
  273. cm && cm->cmsg_len;
  274. cm = CMSG_NXTHDR(msg, cm)) {
  275. if (cm->cmsg_level == SOL_SOCKET &&
  276. cm->cmsg_type == SCM_TIMESTAMPING) {
  277. tss = (void *) CMSG_DATA(cm);
  278. } else if ((cm->cmsg_level == SOL_IP &&
  279. cm->cmsg_type == IP_RECVERR) ||
  280. (cm->cmsg_level == SOL_IPV6 &&
  281. cm->cmsg_type == IPV6_RECVERR) ||
  282. (cm->cmsg_level == SOL_PACKET &&
  283. cm->cmsg_type == PACKET_TX_TIMESTAMP)) {
  284. serr = (void *) CMSG_DATA(cm);
  285. if (serr->ee_errno != ENOMSG ||
  286. serr->ee_origin != SO_EE_ORIGIN_TIMESTAMPING) {
  287. fprintf(stderr, "unknown ip error %d %d\n",
  288. serr->ee_errno,
  289. serr->ee_origin);
  290. serr = NULL;
  291. }
  292. } else if (cm->cmsg_level == SOL_IP &&
  293. cm->cmsg_type == IP_PKTINFO) {
  294. struct in_pktinfo *info = (void *) CMSG_DATA(cm);
  295. print_pktinfo(AF_INET, info->ipi_ifindex,
  296. &info->ipi_spec_dst, &info->ipi_addr);
  297. } else if (cm->cmsg_level == SOL_IPV6 &&
  298. cm->cmsg_type == IPV6_PKTINFO) {
  299. struct in6_pktinfo *info6 = (void *) CMSG_DATA(cm);
  300. print_pktinfo(AF_INET6, info6->ipi6_ifindex,
  301. NULL, &info6->ipi6_addr);
  302. } else
  303. fprintf(stderr, "unknown cmsg %d,%d\n",
  304. cm->cmsg_level, cm->cmsg_type);
  305. if (serr && tss) {
  306. print_timestamp(tss, serr->ee_info, serr->ee_data,
  307. payload_len);
  308. serr = NULL;
  309. tss = NULL;
  310. batch++;
  311. }
  312. }
  313. if (batch > 1) {
  314. fprintf(stderr, "batched %d timestamps\n", batch);
  315. } else if (!batch) {
  316. fprintf(stderr, "Failed to report timestamps\n");
  317. test_failed = true;
  318. }
  319. }
  320. static int recv_errmsg(int fd)
  321. {
  322. static char ctrl[1024 /* overprovision*/];
  323. static struct msghdr msg;
  324. struct iovec entry;
  325. static char *data;
  326. int ret = 0;
  327. data = malloc(cfg_payload_len);
  328. if (!data)
  329. error(1, 0, "malloc");
  330. memset(&msg, 0, sizeof(msg));
  331. memset(&entry, 0, sizeof(entry));
  332. memset(ctrl, 0, sizeof(ctrl));
  333. entry.iov_base = data;
  334. entry.iov_len = cfg_payload_len;
  335. msg.msg_iov = &entry;
  336. msg.msg_iovlen = 1;
  337. msg.msg_name = NULL;
  338. msg.msg_namelen = 0;
  339. msg.msg_control = ctrl;
  340. msg.msg_controllen = sizeof(ctrl);
  341. ret = recvmsg(fd, &msg, MSG_ERRQUEUE);
  342. if (ret == -1 && errno != EAGAIN)
  343. error(1, errno, "recvmsg");
  344. if (ret >= 0) {
  345. __recv_errmsg_cmsg(&msg, ret);
  346. if (cfg_show_payload)
  347. print_payload(data, cfg_payload_len);
  348. }
  349. free(data);
  350. return ret == -1;
  351. }
  352. static uint16_t get_ip_csum(const uint16_t *start, int num_words,
  353. unsigned long sum)
  354. {
  355. int i;
  356. for (i = 0; i < num_words; i++)
  357. sum += start[i];
  358. while (sum >> 16)
  359. sum = (sum & 0xFFFF) + (sum >> 16);
  360. return ~sum;
  361. }
  362. static uint16_t get_udp_csum(const struct udphdr *udph, int alen)
  363. {
  364. unsigned long pseudo_sum, csum_len;
  365. const void *csum_start = udph;
  366. pseudo_sum = htons(IPPROTO_UDP);
  367. pseudo_sum += udph->len;
  368. /* checksum ip(v6) addresses + udp header + payload */
  369. csum_start -= alen * 2;
  370. csum_len = ntohs(udph->len) + alen * 2;
  371. return get_ip_csum(csum_start, csum_len >> 1, pseudo_sum);
  372. }
  373. static int fill_header_ipv4(void *p)
  374. {
  375. struct iphdr *iph = p;
  376. memset(iph, 0, sizeof(*iph));
  377. iph->ihl = 5;
  378. iph->version = 4;
  379. iph->ttl = 2;
  380. iph->saddr = daddr.sin_addr.s_addr; /* set for udp csum calc */
  381. iph->daddr = daddr.sin_addr.s_addr;
  382. iph->protocol = IPPROTO_UDP;
  383. /* kernel writes saddr, csum, len */
  384. return sizeof(*iph);
  385. }
  386. static int fill_header_ipv6(void *p)
  387. {
  388. struct ipv6hdr *ip6h = p;
  389. memset(ip6h, 0, sizeof(*ip6h));
  390. ip6h->version = 6;
  391. ip6h->payload_len = htons(sizeof(struct udphdr) + cfg_payload_len);
  392. ip6h->nexthdr = IPPROTO_UDP;
  393. ip6h->hop_limit = 64;
  394. ip6h->saddr = daddr6.sin6_addr;
  395. ip6h->daddr = daddr6.sin6_addr;
  396. /* kernel does not write saddr in case of ipv6 */
  397. return sizeof(*ip6h);
  398. }
  399. static void fill_header_udp(void *p, bool is_ipv4)
  400. {
  401. struct udphdr *udph = p;
  402. udph->source = ntohs(dest_port + 1); /* spoof */
  403. udph->dest = ntohs(dest_port);
  404. udph->len = ntohs(sizeof(*udph) + cfg_payload_len);
  405. udph->check = 0;
  406. udph->check = get_udp_csum(udph, is_ipv4 ? sizeof(struct in_addr) :
  407. sizeof(struct in6_addr));
  408. }
  409. static void do_test(int family, unsigned int report_opt)
  410. {
  411. char control[2 * CMSG_SPACE(sizeof(uint32_t))];
  412. struct sockaddr_ll laddr;
  413. unsigned int sock_opt;
  414. struct cmsghdr *cmsg;
  415. struct msghdr msg;
  416. struct iovec iov;
  417. char *buf;
  418. int fd, i, val = 1, total_len, epfd = 0;
  419. init_timing_event(&usr_enq);
  420. init_timing_event(&usr_snd);
  421. init_timing_event(&usr_ack);
  422. total_len = cfg_payload_len;
  423. if (cfg_use_pf_packet || cfg_proto == SOCK_RAW) {
  424. total_len += sizeof(struct udphdr);
  425. if (cfg_use_pf_packet || cfg_ipproto == IPPROTO_RAW) {
  426. if (family == PF_INET)
  427. total_len += sizeof(struct iphdr);
  428. else
  429. total_len += sizeof(struct ipv6hdr);
  430. }
  431. /* special case, only rawv6_sendmsg:
  432. * pass proto in sin6_port if not connected
  433. * also see ANK comment in net/ipv4/raw.c
  434. */
  435. daddr6.sin6_port = htons(cfg_ipproto);
  436. }
  437. buf = malloc(total_len);
  438. if (!buf)
  439. error(1, 0, "malloc");
  440. fd = socket(cfg_use_pf_packet ? PF_PACKET : family,
  441. cfg_proto, cfg_ipproto);
  442. if (fd < 0)
  443. error(1, errno, "socket");
  444. if (cfg_use_epoll) {
  445. struct epoll_event ev;
  446. memset(&ev, 0, sizeof(ev));
  447. ev.data.fd = fd;
  448. if (cfg_epollet)
  449. ev.events |= EPOLLET;
  450. epfd = epoll_create(1);
  451. if (epfd <= 0)
  452. error(1, errno, "epoll_create");
  453. if (epoll_ctl(epfd, EPOLL_CTL_ADD, fd, &ev))
  454. error(1, errno, "epoll_ctl");
  455. }
  456. /* reset expected key on each new socket */
  457. saved_tskey = -1;
  458. if (cfg_proto == SOCK_STREAM) {
  459. if (setsockopt(fd, IPPROTO_TCP, TCP_NODELAY,
  460. (char*) &val, sizeof(val)))
  461. error(1, 0, "setsockopt no nagle");
  462. if (family == PF_INET) {
  463. if (connect(fd, (void *) &daddr, sizeof(daddr)))
  464. error(1, errno, "connect ipv4");
  465. } else {
  466. if (connect(fd, (void *) &daddr6, sizeof(daddr6)))
  467. error(1, errno, "connect ipv6");
  468. }
  469. }
  470. if (cfg_do_pktinfo) {
  471. if (family == AF_INET6) {
  472. if (setsockopt(fd, SOL_IPV6, IPV6_RECVPKTINFO,
  473. &val, sizeof(val)))
  474. error(1, errno, "setsockopt pktinfo ipv6");
  475. } else {
  476. if (setsockopt(fd, SOL_IP, IP_PKTINFO,
  477. &val, sizeof(val)))
  478. error(1, errno, "setsockopt pktinfo ipv4");
  479. }
  480. }
  481. sock_opt = SOF_TIMESTAMPING_SOFTWARE |
  482. SOF_TIMESTAMPING_OPT_CMSG |
  483. SOF_TIMESTAMPING_OPT_ID;
  484. if (!cfg_use_cmsg)
  485. sock_opt |= report_opt;
  486. if (cfg_loop_nodata)
  487. sock_opt |= SOF_TIMESTAMPING_OPT_TSONLY;
  488. if (setsockopt(fd, SOL_SOCKET, SO_TIMESTAMPING,
  489. (char *) &sock_opt, sizeof(sock_opt)))
  490. error(1, 0, "setsockopt timestamping");
  491. for (i = 0; i < cfg_num_pkts; i++) {
  492. memset(&msg, 0, sizeof(msg));
  493. memset(buf, 'a' + i, total_len);
  494. if (cfg_use_pf_packet || cfg_proto == SOCK_RAW) {
  495. int off = 0;
  496. if (cfg_use_pf_packet || cfg_ipproto == IPPROTO_RAW) {
  497. if (family == PF_INET)
  498. off = fill_header_ipv4(buf);
  499. else
  500. off = fill_header_ipv6(buf);
  501. }
  502. fill_header_udp(buf + off, family == PF_INET);
  503. }
  504. iov.iov_base = buf;
  505. iov.iov_len = total_len;
  506. if (cfg_proto != SOCK_STREAM) {
  507. if (cfg_use_pf_packet) {
  508. memset(&laddr, 0, sizeof(laddr));
  509. laddr.sll_family = AF_PACKET;
  510. laddr.sll_ifindex = 1;
  511. laddr.sll_protocol = htons(family == AF_INET ? ETH_P_IP : ETH_P_IPV6);
  512. laddr.sll_halen = ETH_ALEN;
  513. msg.msg_name = (void *)&laddr;
  514. msg.msg_namelen = sizeof(laddr);
  515. } else if (family == PF_INET) {
  516. msg.msg_name = (void *)&daddr;
  517. msg.msg_namelen = sizeof(daddr);
  518. } else {
  519. msg.msg_name = (void *)&daddr6;
  520. msg.msg_namelen = sizeof(daddr6);
  521. }
  522. }
  523. msg.msg_iov = &iov;
  524. msg.msg_iovlen = 1;
  525. if (cfg_use_cmsg || cfg_use_cmsg_opt_id) {
  526. memset(control, 0, sizeof(control));
  527. msg.msg_control = control;
  528. msg.msg_controllen = cfg_use_cmsg * CMSG_SPACE(sizeof(uint32_t));
  529. msg.msg_controllen += cfg_use_cmsg_opt_id * CMSG_SPACE(sizeof(uint32_t));
  530. cmsg = NULL;
  531. if (cfg_use_cmsg) {
  532. cmsg = CMSG_FIRSTHDR(&msg);
  533. cmsg->cmsg_level = SOL_SOCKET;
  534. cmsg->cmsg_type = SO_TIMESTAMPING;
  535. cmsg->cmsg_len = CMSG_LEN(sizeof(uint32_t));
  536. *((uint32_t *)CMSG_DATA(cmsg)) = report_opt;
  537. }
  538. if (cfg_use_cmsg_opt_id) {
  539. cmsg = cmsg ? CMSG_NXTHDR(&msg, cmsg) : CMSG_FIRSTHDR(&msg);
  540. cmsg->cmsg_level = SOL_SOCKET;
  541. cmsg->cmsg_type = SCM_TS_OPT_ID;
  542. cmsg->cmsg_len = CMSG_LEN(sizeof(uint32_t));
  543. *((uint32_t *)CMSG_DATA(cmsg)) = ts_opt_id;
  544. saved_tskey = ts_opt_id;
  545. }
  546. }
  547. record_timestamp_usr();
  548. val = sendmsg(fd, &msg, 0);
  549. if (val != total_len)
  550. error(1, errno, "send");
  551. __print_timestamp(" USR", &ts_usr, 0, 0);
  552. /* wait for all errors to be queued, else ACKs arrive OOO */
  553. if (cfg_sleep_usec)
  554. usleep(cfg_sleep_usec);
  555. if (!cfg_busy_poll) {
  556. if (cfg_use_epoll)
  557. __epoll(epfd);
  558. else
  559. __poll(fd);
  560. }
  561. while (!recv_errmsg(fd)) {}
  562. }
  563. print_timing_event("USR-ENQ", &usr_enq);
  564. print_timing_event("USR-SND", &usr_snd);
  565. print_timing_event("USR-ACK", &usr_ack);
  566. if (close(fd))
  567. error(1, errno, "close");
  568. free(buf);
  569. usleep(100 * NSEC_PER_USEC);
  570. }
  571. static void __attribute__((noreturn)) usage(const char *filepath)
  572. {
  573. fprintf(stderr, "\nUsage: %s [options] hostname\n"
  574. "\nwhere options are:\n"
  575. " -4: only IPv4\n"
  576. " -6: only IPv6\n"
  577. " -h: show this message\n"
  578. " -b: busy poll to read from error queue\n"
  579. " -c N: number of packets for each test\n"
  580. " -C: use cmsg to set tstamp recording options\n"
  581. " -e: use level-triggered epoll() instead of poll()\n"
  582. " -E: use event-triggered epoll() instead of poll()\n"
  583. " -F: poll()/epoll() waits forever for an event\n"
  584. " -I: request PKTINFO\n"
  585. " -l N: send N bytes at a time\n"
  586. " -L listen on hostname and port\n"
  587. " -n: set no-payload option\n"
  588. " -N: print timestamps and durations in nsec (instead of usec)\n"
  589. " -o N: use SCM_TS_OPT_ID control message to provide N as tskey\n"
  590. " -p N: connect to port N\n"
  591. " -P: use PF_PACKET\n"
  592. " -r: use raw\n"
  593. " -R: use raw (IP_HDRINCL)\n"
  594. " -S N: usec to sleep before reading error queue\n"
  595. " -t N: tolerance (usec) for timestamp validation\n"
  596. " -u: use udp\n"
  597. " -v: validate SND delay (usec)\n"
  598. " -V: validate ACK delay (usec)\n"
  599. " -x: show payload (up to 70 bytes)\n",
  600. filepath);
  601. exit(1);
  602. }
  603. static void parse_opt(int argc, char **argv)
  604. {
  605. int proto_count = 0;
  606. int c;
  607. while ((c = getopt(argc, argv,
  608. "46bc:CeEFhIl:LnNo:p:PrRS:t:uv:V:x")) != -1) {
  609. switch (c) {
  610. case '4':
  611. do_ipv6 = 0;
  612. break;
  613. case '6':
  614. do_ipv4 = 0;
  615. break;
  616. case 'b':
  617. cfg_busy_poll = true;
  618. break;
  619. case 'c':
  620. cfg_num_pkts = strtoul(optarg, NULL, 10);
  621. break;
  622. case 'C':
  623. cfg_use_cmsg = true;
  624. break;
  625. case 'e':
  626. cfg_use_epoll = true;
  627. break;
  628. case 'E':
  629. cfg_use_epoll = true;
  630. cfg_epollet = true;
  631. case 'F':
  632. cfg_poll_timeout = -1;
  633. break;
  634. case 'I':
  635. cfg_do_pktinfo = true;
  636. break;
  637. case 'l':
  638. cfg_payload_len = strtoul(optarg, NULL, 10);
  639. break;
  640. case 'L':
  641. cfg_do_listen = true;
  642. break;
  643. case 'n':
  644. cfg_loop_nodata = true;
  645. break;
  646. case 'N':
  647. cfg_print_nsec = true;
  648. break;
  649. case 'o':
  650. ts_opt_id = strtoul(optarg, NULL, 10);
  651. cfg_use_cmsg_opt_id = true;
  652. break;
  653. case 'p':
  654. dest_port = strtoul(optarg, NULL, 10);
  655. break;
  656. case 'P':
  657. proto_count++;
  658. cfg_use_pf_packet = true;
  659. cfg_proto = SOCK_DGRAM;
  660. cfg_ipproto = 0;
  661. break;
  662. case 'r':
  663. proto_count++;
  664. cfg_proto = SOCK_RAW;
  665. cfg_ipproto = IPPROTO_UDP;
  666. break;
  667. case 'R':
  668. proto_count++;
  669. cfg_proto = SOCK_RAW;
  670. cfg_ipproto = IPPROTO_RAW;
  671. break;
  672. case 'S':
  673. cfg_sleep_usec = strtoul(optarg, NULL, 10);
  674. break;
  675. case 't':
  676. cfg_delay_tolerance_usec = strtoul(optarg, NULL, 10);
  677. break;
  678. case 'u':
  679. proto_count++;
  680. cfg_proto = SOCK_DGRAM;
  681. cfg_ipproto = IPPROTO_UDP;
  682. break;
  683. case 'v':
  684. cfg_delay_snd = strtoul(optarg, NULL, 10);
  685. break;
  686. case 'V':
  687. cfg_delay_ack = strtoul(optarg, NULL, 10);
  688. break;
  689. case 'x':
  690. cfg_show_payload = true;
  691. break;
  692. case 'h':
  693. default:
  694. usage(argv[0]);
  695. }
  696. }
  697. if (!cfg_payload_len)
  698. error(1, 0, "payload may not be nonzero");
  699. if (cfg_proto != SOCK_STREAM && cfg_payload_len > 1472)
  700. error(1, 0, "udp packet might exceed expected MTU");
  701. if (!do_ipv4 && !do_ipv6)
  702. error(1, 0, "pass -4 or -6, not both");
  703. if (proto_count > 1)
  704. error(1, 0, "pass -P, -r, -R or -u, not multiple");
  705. if (cfg_do_pktinfo && cfg_use_pf_packet)
  706. error(1, 0, "cannot ask for pktinfo over pf_packet");
  707. if (cfg_busy_poll && cfg_use_epoll)
  708. error(1, 0, "pass epoll or busy_poll, not both");
  709. if (cfg_proto == SOCK_STREAM && cfg_use_cmsg_opt_id)
  710. error(1, 0, "TCP sockets don't support SCM_TS_OPT_ID");
  711. if (optind != argc - 1)
  712. error(1, 0, "missing required hostname argument");
  713. }
  714. static void resolve_hostname(const char *hostname)
  715. {
  716. struct addrinfo hints = { .ai_family = do_ipv4 ? AF_INET : AF_INET6 };
  717. struct addrinfo *addrs, *cur;
  718. int have_ipv4 = 0, have_ipv6 = 0;
  719. retry:
  720. if (getaddrinfo(hostname, NULL, &hints, &addrs))
  721. error(1, errno, "getaddrinfo");
  722. cur = addrs;
  723. while (cur && !have_ipv4 && !have_ipv6) {
  724. if (!have_ipv4 && cur->ai_family == AF_INET) {
  725. memcpy(&daddr, cur->ai_addr, sizeof(daddr));
  726. daddr.sin_port = htons(dest_port);
  727. have_ipv4 = 1;
  728. }
  729. else if (!have_ipv6 && cur->ai_family == AF_INET6) {
  730. memcpy(&daddr6, cur->ai_addr, sizeof(daddr6));
  731. daddr6.sin6_port = htons(dest_port);
  732. have_ipv6 = 1;
  733. }
  734. cur = cur->ai_next;
  735. }
  736. if (addrs)
  737. freeaddrinfo(addrs);
  738. if (do_ipv6 && hints.ai_family != AF_INET6) {
  739. hints.ai_family = AF_INET6;
  740. goto retry;
  741. }
  742. do_ipv4 &= have_ipv4;
  743. do_ipv6 &= have_ipv6;
  744. }
  745. static void do_listen(int family, void *addr, int alen)
  746. {
  747. int fd, type;
  748. type = cfg_proto == SOCK_RAW ? SOCK_DGRAM : cfg_proto;
  749. fd = socket(family, type, 0);
  750. if (fd == -1)
  751. error(1, errno, "socket rx");
  752. if (bind(fd, addr, alen))
  753. error(1, errno, "bind rx");
  754. if (type == SOCK_STREAM && listen(fd, 10))
  755. error(1, errno, "listen rx");
  756. /* leave fd open, will be closed on process exit.
  757. * this enables connect() to succeed and avoids icmp replies
  758. */
  759. }
  760. static void do_main(int family)
  761. {
  762. fprintf(stderr, "family: %s %s\n",
  763. family == PF_INET ? "INET" : "INET6",
  764. cfg_use_pf_packet ? "(PF_PACKET)" : "");
  765. fprintf(stderr, "test SND\n");
  766. do_test(family, SOF_TIMESTAMPING_TX_SOFTWARE);
  767. fprintf(stderr, "test ENQ\n");
  768. do_test(family, SOF_TIMESTAMPING_TX_SCHED);
  769. fprintf(stderr, "test ENQ + SND\n");
  770. do_test(family, SOF_TIMESTAMPING_TX_SCHED |
  771. SOF_TIMESTAMPING_TX_SOFTWARE);
  772. if (cfg_proto == SOCK_STREAM) {
  773. fprintf(stderr, "\ntest ACK\n");
  774. do_test(family, SOF_TIMESTAMPING_TX_ACK);
  775. fprintf(stderr, "\ntest SND + ACK\n");
  776. do_test(family, SOF_TIMESTAMPING_TX_SOFTWARE |
  777. SOF_TIMESTAMPING_TX_ACK);
  778. fprintf(stderr, "\ntest ENQ + SND + ACK\n");
  779. do_test(family, SOF_TIMESTAMPING_TX_SCHED |
  780. SOF_TIMESTAMPING_TX_SOFTWARE |
  781. SOF_TIMESTAMPING_TX_ACK);
  782. }
  783. }
  784. const char *sock_names[] = { NULL, "TCP", "UDP", "RAW" };
  785. int main(int argc, char **argv)
  786. {
  787. if (argc == 1)
  788. usage(argv[0]);
  789. parse_opt(argc, argv);
  790. resolve_hostname(argv[argc - 1]);
  791. fprintf(stderr, "protocol: %s\n", sock_names[cfg_proto]);
  792. fprintf(stderr, "payload: %u\n", cfg_payload_len);
  793. fprintf(stderr, "server port: %u\n", dest_port);
  794. fprintf(stderr, "\n");
  795. if (do_ipv4) {
  796. if (cfg_do_listen)
  797. do_listen(PF_INET, &daddr, sizeof(daddr));
  798. do_main(PF_INET);
  799. }
  800. if (do_ipv6) {
  801. if (cfg_do_listen)
  802. do_listen(PF_INET6, &daddr6, sizeof(daddr6));
  803. do_main(PF_INET6);
  804. }
  805. return test_failed;
  806. }