/* * Copyright (c) 2017 Cisco and/or its affiliates. * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at: * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include <stdio.h> #include <sys/types.h> #include <sys/socket.h> #include <netinet/in.h> #include <arpa/inet.h> #include <netdb.h> #include <vppinfra/format.h> #include <signal.h> #include <sys/ucontext.h> #include <sys/time.h> volatile int signal_received; static void unix_signal_handler (int signum, siginfo_t * si, ucontext_t * uc) { signal_received = 1; } static void setup_signal_handler (void) { uword i; struct sigaction sa; for (i = 1; i < 32; i++) { memset (&sa, 0, sizeof (sa)); sa.sa_sigaction = (void *) unix_signal_handler; sa.sa_flags = SA_SIGINFO; switch (i) { /* these signals take the default action */ case SIGABRT: case SIGKILL: case SIGSTOP: case SIGUSR1: case SIGUSR2: continue; /* ignore SIGPIPE, SIGCHLD */ case SIGPIPE: case SIGCHLD: sa.sa_sigaction = (void *) SIG_IGN; break; /* catch and handle all other signals */ default: break; } if (sigaction (i, &sa, 0) < 0) clib_unix_warning ("sigaction %U", format_signal, i); } } int main (int argc, char *argv[]) { int sockfd, portno, n, sent, accfd, reuse; socklen_t client_addr_len; struct sockaddr_in serv_addr; struct sockaddr_in client; struct hostent *server; u8 *rx_buffer = 0, no_echo = 0; struct timeval start, end; long rcvd = 0; double deltat; if (argc > 1 && argc < 3) { fformat (stderr, "usage %s host port\n", argv[0]); exit (0); } if (argc >= 4) { no_echo = atoi (argv[3]); portno = atoi (argv[2]); server = gethostbyname (argv[1]); if (server == NULL) { clib_unix_warning ("gethostbyname"); exit (1); } } else { /* Defaults */ portno = 1234; server = gethostbyname ("6.0.1.1"); if (server == NULL) { clib_unix_warning ("gethostbyname"); exit (1); } } setup_signal_handler (); sockfd = socket (AF_INET, SOCK_STREAM, 0); if (sockfd < 0) { clib_unix_error ("socket"); exit (1); } reuse = 1; if (setsockopt (sockfd, SOL_SOCKET, SO_REUSEADDR, (const char *) &reuse, sizeof (reuse)) < 0) { clib_unix_error ("setsockopt(SO_REUSEADDR) failed"); exit (1); } bzero ((char *) &serv_addr, sizeof (serv_addr)); serv_addr.sin_family = AF_INET; bcopy ((char *) server->h_addr, (char *) &serv_addr.sin_addr.s_addr, server->h_length); serv_addr.sin_port = htons (portno); if (bind (sockfd, (const void *) &serv_addr, sizeof (serv_addr)) < 0) { clib_unix_warning ("bind"); exit (1); } vec_validate (rx_buffer, 128 << 10); if (listen (sockfd, 5 /* backlog */ ) < 0) { clib_unix_warning ("listen"); close (sockfd); return 1; } while (1) { if (signal_received) break; client_addr_len = sizeof (struct sockaddr); accfd = accept (sockfd, (struct sockaddr *) &client, &client_addr_len); if (accfd < 0) { clib_unix_warning ("accept"); continue; } fformat (stderr, "Accepted connection from: %s : %d\n", inet_ntoa (client.sin_addr), client.sin_port); gettimeofday (&start, NULL); while (1) { n = recv (accfd, rx_buffer, vec_len (rx_buffer), 0 /* flags */ ); if (n == 0) { /* Graceful exit */ close (accfd); gettimeofday (&end, NULL); deltat = (end.tv_sec - start.tv_sec); deltat += (end.tv_usec - start.tv_usec) / 1000000.0; clib_warning ("Finished in %.6f", deltat); clib_warning ("%.4f Gbit/second %s", (((f64) rcvd * 8.0) / deltat / 1e9), no_echo ? "half" : "full"); rcvd = 0; break; } if (n < 0) { clib_unix_warning ("recv"); close (accfd); break; } if (signal_received) break; rcvd += n; if (no_echo) continue; sent = send (accfd, rx_buffer, n, 0 /* flags */ ); if (n < 0) { clib_unix_warning ("send"); close (accfd); break; } if (sent != n) { clib_warning ("sent %d not %d", sent, n); } if (signal_received) break; } } close (sockfd); return 0; } /* * fd.io coding-style-patch-verification: ON * * Local Variables: * eval: (c-set-style "gnu") * End: */