#include <arpa/inet.h> #include <linux/if.h> #include <netinet/in.h> #include <stdlib.h> #include <sys/ioctl.h> #include <sys/socket.h> #include <unistd.h> #include <atomic> #include <mutex> #include <thread> #include "utils/RWLock.h" // Defined only in ifc_utils.c, in the kernel, and in the NDK. #ifndef SIOCKILLADDR #define SIOCKILLADDR 0x8939 #endif #ifndef TCP_LINGER2 #define TCP_LINGER2 8 #endif #define KILL_INTERVAL_MS 10 #define CONNECT_THREADS 1 #define PERROR_EXIT(msg) { do { perror((msg)); exit(1); } while (0); }; // Ensures that sockets don't stay in TIME_WAIT state. void setSoLinger(int s) { const struct linger l = { 0, // off 0, // 0 seconds }; if (setsockopt(s, SOL_SOCKET, SO_LINGER, &l, sizeof(l)) == -1) { PERROR_EXIT("SO_LINGER"); } const int nolinger = -1; if (setsockopt(s, SOL_TCP, TCP_LINGER2, &nolinger, sizeof(nolinger)) == -1) { PERROR_EXIT("TCP_LINGER2"); } } // Binds to a random port on a random loopback address. We don't just use 127.0.0.1 because we don't // want this test to kill unrelated connections on loopback. int bindRandomAddr() { sockaddr_in sin; sin.sin_family = AF_INET; sin.sin_port = 0; sin.sin_addr.s_addr = htonl(INADDR_LOOPBACK); while (sin.sin_addr.s_addr == htonl(INADDR_LOOPBACK)) { arc4random_buf( ((uint8_t *) &sin.sin_addr.s_addr) + 1, sizeof(sin.sin_addr.s_addr) - 1); } int listensock; if ((listensock = socket(AF_INET, SOCK_STREAM, 0)) == -1) PERROR_EXIT("listensock"); if (bind(listensock, (sockaddr *) &sin, sizeof(sin)) == -1) PERROR_EXIT("bind"); if (listen(listensock, 10) == -1) PERROR_EXIT("listen"); return listensock; } // Thread that calls SIOCKILLADDR in a loop. void killSockets(sockaddr_in listenaddr, int intervalMs, android::RWLock *lock) { ifreq ifr; memset(&ifr, 0, sizeof(ifr)); listenaddr.sin_port = 0; strncpy(ifr.ifr_name, "lo", strlen("lo")); memcpy(&ifr.ifr_addr, &listenaddr, sizeof(listenaddr)); int ioctlsock = socket(AF_INET, SOCK_DGRAM, 0); if (ioctlsock == -1) PERROR_EXIT("ioctlsock"); while(true) { lock->writeLock(); if (ioctl(ioctlsock, SIOCKILLADDR, &ifr) != 0) { PERROR_EXIT("SIOCKILLADDR failed, did you run 32-bit userspace on a 64-bit kernel?"); } lock->unlock(); std::this_thread::sleep_for(std::chrono::milliseconds(intervalMs)); } } // Thread that calls connect() in a loop. void connectLoop(sockaddr_in listenaddr, int listensock, android::RWLock *lock, std::atomic<unsigned int> *attempts) { while(true) { int s = socket(AF_INET, SOCK_STREAM, 0); setSoLinger(s); // Don't call SIOCKILLADDR while connect() is running, or we'll end up with lots of // connections in state FIN_WAITx or TIME_WAIT, which will then slow down future // due to SYN retransmits. lock->readLock(); if (connect(s, (sockaddr *) &listenaddr, sizeof(listenaddr)) == -1) PERROR_EXIT("connect"); lock->unlock(); send(s, "foo", 3, 0); int acceptedsock = accept(listensock, NULL, 0); if (close(acceptedsock) == -1) PERROR_EXIT("close"); if (close(s) == -1) PERROR_EXIT("close"); attempts->fetch_add(1); } } // Thread that prints progress every second. void progressThread(std::atomic<unsigned int> *attempts) { uint32_t elapsed = 0; uint32_t total, previous = 0; while (true) { std::this_thread::sleep_for(std::chrono::seconds(1)); elapsed++; total = attempts->load(); printf("%ds: %u cps, total %u\n", elapsed, total-previous, total); fflush(stdout); previous = total; } } int main() { int listensock = bindRandomAddr(); struct sockaddr_in sin; socklen_t len = sizeof(sin); if (getsockname(listensock, (sockaddr *) &sin, &len) == -1) PERROR_EXIT("getsockname"); printf("Using address %s:%d\n", inet_ntoa(sin.sin_addr), ntohs(sin.sin_port)); android::RWLock lock; std::atomic<unsigned int> attempts; attempts.store(0); std::thread t0(killSockets, sin, KILL_INTERVAL_MS, &lock); for (size_t i = 0; i < CONNECT_THREADS; i++) { std::thread(connectLoop, sin, listensock, &lock, &attempts).detach(); } progressThread(&attempts); return 0; }