blob: 661ea7e4a3505fd6e0fc234bcb6702c1ee0b1b8d [file] [log] [blame]
John Fastabend69e8cc12017-08-15 22:33:32 -07001/* Copyright (c) 2017 Covalent IO, Inc. http://covalent.io
2 *
3 * This program is free software; you can redistribute it and/or
4 * modify it under the terms of version 2 of the GNU General Public
5 * License as published by the Free Software Foundation.
6 *
7 * This program is distributed in the hope that it will be useful, but
8 * WITHOUT ANY WARRANTY; without even the implied warranty of
9 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
10 * General Public License for more details.
11 */
12#include <stdio.h>
13#include <stdlib.h>
14#include <sys/socket.h>
15#include <sys/ioctl.h>
16#include <sys/select.h>
17#include <netinet/in.h>
18#include <arpa/inet.h>
19#include <unistd.h>
20#include <string.h>
21#include <errno.h>
22#include <sys/ioctl.h>
23#include <stdbool.h>
24#include <signal.h>
25#include <fcntl.h>
John Fastabendd7d64372018-01-22 10:36:02 -080026#include <sys/wait.h>
John Fastabend66fdd1a2018-01-22 10:36:19 -080027#include <time.h>
John Fastabend69e8cc12017-08-15 22:33:32 -070028
29#include <sys/time.h>
30#include <sys/types.h>
31
32#include <linux/netlink.h>
33#include <linux/socket.h>
34#include <linux/sock_diag.h>
35#include <linux/bpf.h>
36#include <linux/if_link.h>
37#include <assert.h>
38#include <libgen.h>
39
John Fastabend6627426f2018-01-22 10:35:27 -080040#include <getopt.h>
41
John Fastabend69e8cc12017-08-15 22:33:32 -070042#include "../bpf/bpf_load.h"
43#include "../bpf/bpf_util.h"
44#include "../bpf/libbpf.h"
45
46int running;
47void running_handler(int a);
48
49/* randomly selected ports for testing on lo */
50#define S1_PORT 10000
51#define S2_PORT 10001
52
John Fastabend6627426f2018-01-22 10:35:27 -080053/* global sockets */
54int s1, s2, c1, c2, p1, p2;
55
56static const struct option long_options[] = {
57 {"help", no_argument, NULL, 'h' },
58 {"cgroup", required_argument, NULL, 'c' },
59 {"rate", required_argument, NULL, 'r' },
60 {"verbose", no_argument, NULL, 'v' },
John Fastabendeaf8c6e2018-01-22 10:35:45 -080061 {"iov_count", required_argument, NULL, 'i' },
62 {"length", required_argument, NULL, 'l' },
63 {"test", required_argument, NULL, 't' },
John Fastabend6627426f2018-01-22 10:35:27 -080064 {0, 0, NULL, 0 }
65};
66
67static void usage(char *argv[])
John Fastabend69e8cc12017-08-15 22:33:32 -070068{
John Fastabend6627426f2018-01-22 10:35:27 -080069 int i;
70
71 printf(" Usage: %s --cgroup <cgroup_path>\n", argv[0]);
72 printf(" options:\n");
73 for (i = 0; long_options[i].name != 0; i++) {
74 printf(" --%-12s", long_options[i].name);
75 if (long_options[i].flag != NULL)
76 printf(" flag (internal value:%d)\n",
77 *long_options[i].flag);
78 else
79 printf(" -%c\n", long_options[i].val);
80 }
81 printf("\n");
82}
83
84static int sockmap_init_sockets(void)
85{
86 int i, err, one = 1;
John Fastabend69e8cc12017-08-15 22:33:32 -070087 struct sockaddr_in addr;
John Fastabend69e8cc12017-08-15 22:33:32 -070088 int *fds[4] = {&s1, &s2, &c1, &c2};
John Fastabend69e8cc12017-08-15 22:33:32 -070089
90 s1 = s2 = p1 = p2 = c1 = c2 = 0;
91
92 /* Init sockets */
93 for (i = 0; i < 4; i++) {
94 *fds[i] = socket(AF_INET, SOCK_STREAM, 0);
95 if (*fds[i] < 0) {
96 perror("socket s1 failed()");
John Fastabend6627426f2018-01-22 10:35:27 -080097 return errno;
John Fastabend69e8cc12017-08-15 22:33:32 -070098 }
99 }
100
101 /* Allow reuse */
102 for (i = 0; i < 2; i++) {
103 err = setsockopt(*fds[i], SOL_SOCKET, SO_REUSEADDR,
104 (char *)&one, sizeof(one));
105 if (err) {
106 perror("setsockopt failed()");
John Fastabend6627426f2018-01-22 10:35:27 -0800107 return errno;
John Fastabend69e8cc12017-08-15 22:33:32 -0700108 }
109 }
110
111 /* Non-blocking sockets */
112 for (i = 0; i < 4; i++) {
113 err = ioctl(*fds[i], FIONBIO, (char *)&one);
114 if (err < 0) {
115 perror("ioctl s1 failed()");
John Fastabend6627426f2018-01-22 10:35:27 -0800116 return errno;
John Fastabend69e8cc12017-08-15 22:33:32 -0700117 }
118 }
119
120 /* Bind server sockets */
121 memset(&addr, 0, sizeof(struct sockaddr_in));
122 addr.sin_family = AF_INET;
123 addr.sin_addr.s_addr = inet_addr("127.0.0.1");
124
125 addr.sin_port = htons(S1_PORT);
126 err = bind(s1, (struct sockaddr *)&addr, sizeof(addr));
127 if (err < 0) {
128 perror("bind s1 failed()\n");
John Fastabend6627426f2018-01-22 10:35:27 -0800129 return errno;
John Fastabend69e8cc12017-08-15 22:33:32 -0700130 }
131
132 addr.sin_port = htons(S2_PORT);
133 err = bind(s2, (struct sockaddr *)&addr, sizeof(addr));
134 if (err < 0) {
135 perror("bind s2 failed()\n");
John Fastabend6627426f2018-01-22 10:35:27 -0800136 return errno;
John Fastabend69e8cc12017-08-15 22:33:32 -0700137 }
138
139 /* Listen server sockets */
140 addr.sin_port = htons(S1_PORT);
141 err = listen(s1, 32);
142 if (err < 0) {
143 perror("listen s1 failed()\n");
John Fastabend6627426f2018-01-22 10:35:27 -0800144 return errno;
John Fastabend69e8cc12017-08-15 22:33:32 -0700145 }
146
147 addr.sin_port = htons(S2_PORT);
148 err = listen(s2, 32);
149 if (err < 0) {
150 perror("listen s1 failed()\n");
John Fastabend6627426f2018-01-22 10:35:27 -0800151 return errno;
John Fastabend69e8cc12017-08-15 22:33:32 -0700152 }
153
154 /* Initiate Connect */
155 addr.sin_port = htons(S1_PORT);
156 err = connect(c1, (struct sockaddr *)&addr, sizeof(addr));
157 if (err < 0 && errno != EINPROGRESS) {
158 perror("connect c1 failed()\n");
John Fastabend6627426f2018-01-22 10:35:27 -0800159 return errno;
John Fastabend69e8cc12017-08-15 22:33:32 -0700160 }
161
162 addr.sin_port = htons(S2_PORT);
163 err = connect(c2, (struct sockaddr *)&addr, sizeof(addr));
164 if (err < 0 && errno != EINPROGRESS) {
165 perror("connect c2 failed()\n");
John Fastabend6627426f2018-01-22 10:35:27 -0800166 return errno;
167 } else if (err < 0) {
168 err = 0;
John Fastabend69e8cc12017-08-15 22:33:32 -0700169 }
170
171 /* Accept Connecrtions */
172 p1 = accept(s1, NULL, NULL);
173 if (p1 < 0) {
174 perror("accept s1 failed()\n");
John Fastabend6627426f2018-01-22 10:35:27 -0800175 return errno;
John Fastabend69e8cc12017-08-15 22:33:32 -0700176 }
177
178 p2 = accept(s2, NULL, NULL);
179 if (p2 < 0) {
180 perror("accept s1 failed()\n");
John Fastabend6627426f2018-01-22 10:35:27 -0800181 return errno;
John Fastabend69e8cc12017-08-15 22:33:32 -0700182 }
183
John Fastabend69e8cc12017-08-15 22:33:32 -0700184 printf("connected sockets: c1 <-> p1, c2 <-> p2\n");
185 printf("cgroups binding: c1(%i) <-> s1(%i) - - - c2(%i) <-> s2(%i)\n",
186 c1, s1, c2, s2);
John Fastabend6627426f2018-01-22 10:35:27 -0800187 return 0;
188}
189
John Fastabendeaf8c6e2018-01-22 10:35:45 -0800190struct msg_stats {
191 size_t bytes_sent;
192 size_t bytes_recvd;
John Fastabend66fdd1a2018-01-22 10:36:19 -0800193 struct timespec start;
194 struct timespec end;
John Fastabendeaf8c6e2018-01-22 10:35:45 -0800195};
196
197static int msg_loop(int fd, int iov_count, int iov_length, int cnt,
198 struct msg_stats *s, bool tx)
199{
200 struct msghdr msg = {0};
John Fastabend66fdd1a2018-01-22 10:36:19 -0800201 int err, i, flags = MSG_NOSIGNAL;
John Fastabendeaf8c6e2018-01-22 10:35:45 -0800202 struct iovec *iov;
John Fastabendeaf8c6e2018-01-22 10:35:45 -0800203
204 iov = calloc(iov_count, sizeof(struct iovec));
205 if (!iov)
206 return errno;
207
208 for (i = 0; i < iov_count; i++) {
209 char *d = calloc(iov_length, sizeof(char));
210
211 if (!d) {
212 fprintf(stderr, "iov_count %i/%i OOM\n", i, iov_count);
213 goto out_errno;
214 }
215 iov[i].iov_base = d;
216 iov[i].iov_len = iov_length;
217 }
218
219 msg.msg_iov = iov;
220 msg.msg_iovlen = iov_count;
221
222 if (tx) {
John Fastabend66fdd1a2018-01-22 10:36:19 -0800223 clock_gettime(CLOCK_MONOTONIC, &s->start);
John Fastabendeaf8c6e2018-01-22 10:35:45 -0800224 for (i = 0; i < cnt; i++) {
225 int sent = sendmsg(fd, &msg, flags);
226
227 if (sent < 0) {
228 perror("send loop error:");
229 goto out_errno;
230 }
231 s->bytes_sent += sent;
232 }
John Fastabend66fdd1a2018-01-22 10:36:19 -0800233 clock_gettime(CLOCK_MONOTONIC, &s->end);
John Fastabendeaf8c6e2018-01-22 10:35:45 -0800234 } else {
235 int slct, recv, max_fd = fd;
236 struct timeval timeout;
237 float total_bytes;
238 fd_set w;
239
240 total_bytes = (float)iov_count * (float)iov_length * (float)cnt;
John Fastabend66fdd1a2018-01-22 10:36:19 -0800241 err = clock_gettime(CLOCK_MONOTONIC, &s->start);
242 if (err < 0)
243 perror("recv start time: ");
John Fastabendeaf8c6e2018-01-22 10:35:45 -0800244 while (s->bytes_recvd < total_bytes) {
245 timeout.tv_sec = 1;
246 timeout.tv_usec = 0;
247
248 /* FD sets */
249 FD_ZERO(&w);
250 FD_SET(fd, &w);
251
252 slct = select(max_fd + 1, &w, NULL, NULL, &timeout);
253 if (slct == -1) {
254 perror("select()");
John Fastabend66fdd1a2018-01-22 10:36:19 -0800255 clock_gettime(CLOCK_MONOTONIC, &s->end);
John Fastabendeaf8c6e2018-01-22 10:35:45 -0800256 goto out_errno;
257 } else if (!slct) {
258 fprintf(stderr, "unexpected timeout\n");
259 errno = -EIO;
John Fastabend66fdd1a2018-01-22 10:36:19 -0800260 clock_gettime(CLOCK_MONOTONIC, &s->end);
John Fastabendeaf8c6e2018-01-22 10:35:45 -0800261 goto out_errno;
262 }
263
264 recv = recvmsg(fd, &msg, flags);
265 if (recv < 0) {
266 if (errno != EWOULDBLOCK) {
John Fastabend66fdd1a2018-01-22 10:36:19 -0800267 clock_gettime(CLOCK_MONOTONIC, &s->end);
John Fastabendeaf8c6e2018-01-22 10:35:45 -0800268 perror("recv failed()\n");
269 goto out_errno;
270 }
271 }
272
273 s->bytes_recvd += recv;
274 }
John Fastabend66fdd1a2018-01-22 10:36:19 -0800275 clock_gettime(CLOCK_MONOTONIC, &s->end);
John Fastabendeaf8c6e2018-01-22 10:35:45 -0800276 }
277
278 for (i = 0; i < iov_count; i++)
279 free(iov[i].iov_base);
280 free(iov);
281 return 0;
282out_errno:
283 for (i = 0; i < iov_count; i++)
284 free(iov[i].iov_base);
285 free(iov);
286 return errno;
287}
288
John Fastabend66fdd1a2018-01-22 10:36:19 -0800289static float giga = 1000000000;
290
291static inline float sentBps(struct msg_stats s)
292{
293 return s.bytes_sent / (s.end.tv_sec - s.start.tv_sec);
294}
295
296static inline float recvdBps(struct msg_stats s)
297{
298 return s.bytes_recvd / (s.end.tv_sec - s.start.tv_sec);
299}
300
John Fastabendeaf8c6e2018-01-22 10:35:45 -0800301static int sendmsg_test(int iov_count, int iov_buf, int cnt, int verbose)
302{
John Fastabendd7d64372018-01-22 10:36:02 -0800303 int txpid, rxpid, err = 0;
John Fastabendeaf8c6e2018-01-22 10:35:45 -0800304 struct msg_stats s = {0};
John Fastabendd7d64372018-01-22 10:36:02 -0800305 int status;
John Fastabend66fdd1a2018-01-22 10:36:19 -0800306 float sent_Bps = 0, recvd_Bps = 0;
John Fastabendeaf8c6e2018-01-22 10:35:45 -0800307
John Fastabendd7d64372018-01-22 10:36:02 -0800308 errno = 0;
309
310 rxpid = fork();
311 if (rxpid == 0) {
312 err = msg_loop(p2, iov_count, iov_buf, cnt, &s, false);
313 if (err)
314 fprintf(stderr,
315 "msg_loop_rx: iov_count %i iov_buf %i cnt %i err %i\n",
316 iov_count, iov_buf, cnt, err);
John Fastabendd7d64372018-01-22 10:36:02 -0800317 shutdown(p2, SHUT_RDWR);
318 shutdown(p1, SHUT_RDWR);
John Fastabend66fdd1a2018-01-22 10:36:19 -0800319 if (s.end.tv_sec - s.start.tv_sec) {
320 sent_Bps = sentBps(s);
321 recvd_Bps = recvdBps(s);
322 }
323 fprintf(stdout,
324 "rx_sendmsg: TX: %zuB %fB/s %fGB/s RX: %zuB %fB/s %fGB/s\n",
325 s.bytes_sent, sent_Bps, sent_Bps/giga,
326 s.bytes_recvd, recvd_Bps, recvd_Bps/giga);
John Fastabendd7d64372018-01-22 10:36:02 -0800327 exit(1);
328 } else if (rxpid == -1) {
329 perror("msg_loop_rx: ");
330 return errno;
John Fastabendeaf8c6e2018-01-22 10:35:45 -0800331 }
332
John Fastabendd7d64372018-01-22 10:36:02 -0800333 txpid = fork();
334 if (txpid == 0) {
335 err = msg_loop(c1, iov_count, iov_buf, cnt, &s, true);
336 if (err)
337 fprintf(stderr,
338 "msg_loop_tx: iov_count %i iov_buf %i cnt %i err %i\n",
339 iov_count, iov_buf, cnt, err);
John Fastabendd7d64372018-01-22 10:36:02 -0800340 shutdown(c1, SHUT_RDWR);
John Fastabend66fdd1a2018-01-22 10:36:19 -0800341 if (s.end.tv_sec - s.start.tv_sec) {
342 sent_Bps = sentBps(s);
343 recvd_Bps = recvdBps(s);
344 }
345 fprintf(stdout,
346 "tx_sendmsg: TX: %zuB %fB/s %f GB/s RX: %zuB %fB/s %fGB/s\n",
347 s.bytes_sent, sent_Bps, sent_Bps/giga,
348 s.bytes_recvd, recvd_Bps, recvd_Bps/giga);
John Fastabendd7d64372018-01-22 10:36:02 -0800349 exit(1);
350 } else if (txpid == -1) {
351 perror("msg_loop_tx: ");
352 return errno;
353 }
John Fastabendeaf8c6e2018-01-22 10:35:45 -0800354
John Fastabendd7d64372018-01-22 10:36:02 -0800355 assert(waitpid(rxpid, &status, 0) == rxpid);
356 assert(waitpid(txpid, &status, 0) == txpid);
John Fastabendeaf8c6e2018-01-22 10:35:45 -0800357 return err;
358}
359
John Fastabend6627426f2018-01-22 10:35:27 -0800360static int forever_ping_pong(int rate, int verbose)
361{
362 struct timeval timeout;
363 char buf[1024] = {0};
364 int sc;
365
366 timeout.tv_sec = 10;
367 timeout.tv_usec = 0;
John Fastabend69e8cc12017-08-15 22:33:32 -0700368
369 /* Ping/Pong data from client to server */
370 sc = send(c1, buf, sizeof(buf), 0);
371 if (sc < 0) {
372 perror("send failed()\n");
John Fastabend6627426f2018-01-22 10:35:27 -0800373 return sc;
John Fastabend69e8cc12017-08-15 22:33:32 -0700374 }
375
376 do {
John Fastabend6627426f2018-01-22 10:35:27 -0800377 int s, rc, i, max_fd = p2;
378 fd_set w;
John Fastabend69e8cc12017-08-15 22:33:32 -0700379
380 /* FD sets */
381 FD_ZERO(&w);
382 FD_SET(c1, &w);
383 FD_SET(c2, &w);
384 FD_SET(p1, &w);
385 FD_SET(p2, &w);
386
387 s = select(max_fd + 1, &w, NULL, NULL, &timeout);
388 if (s == -1) {
389 perror("select()");
390 break;
391 } else if (!s) {
392 fprintf(stderr, "unexpected timeout\n");
393 break;
394 }
395
396 for (i = 0; i <= max_fd && s > 0; ++i) {
397 if (!FD_ISSET(i, &w))
398 continue;
399
400 s--;
401
402 rc = recv(i, buf, sizeof(buf), 0);
403 if (rc < 0) {
404 if (errno != EWOULDBLOCK) {
405 perror("recv failed()\n");
John Fastabend6627426f2018-01-22 10:35:27 -0800406 return rc;
John Fastabend69e8cc12017-08-15 22:33:32 -0700407 }
408 }
409
410 if (rc == 0) {
411 close(i);
412 break;
413 }
414
415 sc = send(i, buf, rc, 0);
416 if (sc < 0) {
417 perror("send failed()\n");
John Fastabend6627426f2018-01-22 10:35:27 -0800418 return sc;
John Fastabend69e8cc12017-08-15 22:33:32 -0700419 }
420 }
John Fastabend6627426f2018-01-22 10:35:27 -0800421
422 if (rate)
423 sleep(rate);
424
425 if (verbose) {
John Fastabend69e8cc12017-08-15 22:33:32 -0700426 printf(".");
427 fflush(stdout);
428
429 }
430 } while (running);
431
John Fastabend6627426f2018-01-22 10:35:27 -0800432 return 0;
John Fastabend69e8cc12017-08-15 22:33:32 -0700433}
434
John Fastabendeaf8c6e2018-01-22 10:35:45 -0800435enum {
436 PING_PONG,
437 SENDMSG,
438};
439
John Fastabend69e8cc12017-08-15 22:33:32 -0700440int main(int argc, char **argv)
441{
John Fastabendeaf8c6e2018-01-22 10:35:45 -0800442 int iov_count = 1, length = 1024, rate = 1, verbose = 0;
John Fastabend6627426f2018-01-22 10:35:27 -0800443 int opt, longindex, err, cg_fd = 0;
John Fastabendeaf8c6e2018-01-22 10:35:45 -0800444 int test = PING_PONG;
John Fastabend69e8cc12017-08-15 22:33:32 -0700445 char filename[256];
John Fastabend69e8cc12017-08-15 22:33:32 -0700446
John Fastabendeaf8c6e2018-01-22 10:35:45 -0800447 while ((opt = getopt_long(argc, argv, "hvc:r:i:l:t:",
John Fastabend6627426f2018-01-22 10:35:27 -0800448 long_options, &longindex)) != -1) {
449 switch (opt) {
450 /* Cgroup configuration */
451 case 'c':
452 cg_fd = open(optarg, O_DIRECTORY, O_RDONLY);
453 if (cg_fd < 0) {
454 fprintf(stderr,
455 "ERROR: (%i) open cg path failed: %s\n",
456 cg_fd, optarg);
457 return cg_fd;
458 }
459 break;
460 case 'r':
461 rate = atoi(optarg);
462 break;
463 case 'v':
464 verbose = 1;
465 break;
John Fastabendeaf8c6e2018-01-22 10:35:45 -0800466 case 'i':
467 iov_count = atoi(optarg);
468 break;
469 case 'l':
470 length = atoi(optarg);
471 break;
472 case 't':
473 if (strcmp(optarg, "ping") == 0) {
474 test = PING_PONG;
475 } else if (strcmp(optarg, "sendmsg") == 0) {
476 test = SENDMSG;
477 } else {
478 usage(argv);
479 return -1;
480 }
481 break;
John Fastabend6627426f2018-01-22 10:35:27 -0800482 case 'h':
483 default:
484 usage(argv);
485 return -1;
486 }
487 }
488
489 if (!cg_fd) {
490 fprintf(stderr, "%s requires cgroup option: --cgroup <path>\n",
491 argv[0]);
492 return -1;
493 }
494
John Fastabend69e8cc12017-08-15 22:33:32 -0700495 snprintf(filename, sizeof(filename), "%s_kern.o", argv[0]);
496
497 running = 1;
498
499 /* catch SIGINT */
500 signal(SIGINT, running_handler);
501
502 if (load_bpf_file(filename)) {
503 fprintf(stderr, "load_bpf_file: (%s) %s\n",
504 filename, strerror(errno));
505 return 1;
506 }
507
John Fastabend69e8cc12017-08-15 22:33:32 -0700508 /* Attach programs to sockmap */
John Fastabend464bc0f2017-08-28 07:10:04 -0700509 err = bpf_prog_attach(prog_fd[0], map_fd[0],
510 BPF_SK_SKB_STREAM_PARSER, 0);
511 if (err) {
512 fprintf(stderr, "ERROR: bpf_prog_attach (sockmap): %d (%s)\n",
513 err, strerror(errno));
514 return err;
515 }
516
517 err = bpf_prog_attach(prog_fd[1], map_fd[0],
518 BPF_SK_SKB_STREAM_VERDICT, 0);
John Fastabend69e8cc12017-08-15 22:33:32 -0700519 if (err) {
520 fprintf(stderr, "ERROR: bpf_prog_attach (sockmap): %d (%s)\n",
521 err, strerror(errno));
522 return err;
523 }
524
525 /* Attach to cgroups */
526 err = bpf_prog_attach(prog_fd[2], cg_fd, BPF_CGROUP_SOCK_OPS, 0);
527 if (err) {
528 fprintf(stderr, "ERROR: bpf_prog_attach (groups): %d (%s)\n",
529 err, strerror(errno));
530 return err;
531 }
532
John Fastabend6627426f2018-01-22 10:35:27 -0800533 err = sockmap_init_sockets();
John Fastabend69e8cc12017-08-15 22:33:32 -0700534 if (err) {
535 fprintf(stderr, "ERROR: test socket failed: %d\n", err);
John Fastabend6627426f2018-01-22 10:35:27 -0800536 goto out;
John Fastabend69e8cc12017-08-15 22:33:32 -0700537 }
John Fastabend6627426f2018-01-22 10:35:27 -0800538
John Fastabendeaf8c6e2018-01-22 10:35:45 -0800539 if (test == PING_PONG)
540 err = forever_ping_pong(rate, verbose);
541 else if (test == SENDMSG)
542 err = sendmsg_test(iov_count, length, rate, verbose);
543 else
544 fprintf(stderr, "unknown test\n");
John Fastabend6627426f2018-01-22 10:35:27 -0800545out:
546 close(s1);
547 close(s2);
548 close(p1);
549 close(p2);
550 close(c1);
551 close(c2);
552 close(cg_fd);
553 return err;
John Fastabend69e8cc12017-08-15 22:33:32 -0700554}
555
556void running_handler(int a)
557{
558 running = 0;
559}