/* * Copyright (c) 2023-2023 Huawei Device Co., Ltd. All rights reserved. * * Redistribution and use in source and binary forms, with or without modification, * are permitted provided that the following conditions are met: * * 1. Redistributions of source code must retain the above copyright notice, this list of * conditions and the following disclaimer. * * 2. Redistributions in binary form must reproduce the above copyright notice, this list * of conditions and the following disclaimer in the documentation and/or other materials * provided with the distribution. * * 3. Neither the name of the copyright holder nor the names of its contributors may be used * to endorse or promote products derived from this software without specific prior written * permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #include #include #include #include #include "It_container_test.h" static const char *NETMASK = "255.255.255.0"; static const char *GW = "192.168.100.1"; static const char *IFNAME = "veth0"; static const char *SERVER_IP = "192.168.100.6"; static const int SERVER_PORT = 8005; static const char *PEER_IP = "192.168.100.5"; static const char *SERVER_MSG = "===Hi, I'm Server.==="; static const char *PEER_MSG = "===Hi, I'm Peer.==="; static const int TRY_COUNT = 5; static const int DATA_LEN = 128; static int TryResetNetaddr(const char *ifname, const char *ip, const char *netmask, const char *gw) { int ret; struct ifreq ifr; struct rtentry rt; int fd = socket(AF_INET, SOCK_DGRAM, IPPROTO_IP); if (fd < 0) { return -1; } ret = strncpy_s(ifr.ifr_name, sizeof(ifr.ifr_name), ifname, IFNAMSIZ); if (ret != EOK) { close(fd); return -1; } ifr.ifr_addr.sa_family = AF_INET; inet_pton(AF_INET, netmask, ifr.ifr_addr.sa_data + 2); /* 2: offset */ ret = ioctl(fd, SIOCSIFNETMASK, &ifr); if (ret == 0) { inet_pton(AF_INET, ip, ifr.ifr_addr.sa_data + 2); /* 2: offset */ ret = ioctl(fd, SIOCSIFADDR, &ifr); if (ret == 0) { struct sockaddr_in* addr = reinterpret_cast(&rt.rt_gateway); addr->sin_family = AF_INET; addr->sin_addr.s_addr = inet_addr(GW); rt.rt_flags = RTF_GATEWAY; ret = ioctl(fd, SIOCADDRT, &rt); } } if (ret != 0) { (void)close(fd); return ret; } ret = close(fd); return ret; } static int ResetNetaddr(const char *ifname, const char *ip, const char *netmask, const char *gw) { int ret; int try_count = TRY_COUNT; while (try_count--) { ret = TryResetNetaddr(ifname, ip, netmask, gw); if (ret == 0) { break; } sleep(1); } return ret; } static int TcpClient(void) { int ret = 0; int peer; int try_count = TRY_COUNT; char recv_data[DATA_LEN]; struct sockaddr_in server_addr; peer = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP); if (peer < 0) { return EXIT_CODE_ERRNO_1; } server_addr.sin_family = AF_INET; server_addr.sin_addr.s_addr = inet_addr(SERVER_IP); server_addr.sin_port = htons(SERVER_PORT); (void)memset_s(&(server_addr.sin_zero), sizeof(server_addr.sin_zero), 0, sizeof(server_addr.sin_zero)); do { sleep(1); ret = connect(peer, const_cast(reinterpret_cast(&server_addr)), sizeof(struct sockaddr)); } while (ret !=0 && (try_count--)); if (ret < 0) { return EXIT_CODE_ERRNO_2; } ret = send(peer, PEER_MSG, strlen(PEER_MSG) + 1, 0); if (ret < 0) { return EXIT_CODE_ERRNO_3; } ret = recv(peer, recv_data, sizeof(recv_data), 0); if (ret < 0) { return EXIT_CODE_ERRNO_4; } ret = strcmp(recv_data, SERVER_MSG); if (ret != 0) { return EXIT_CODE_ERRNO_5; } (void)close(peer); return 0; } static int TcpServer(void) { int ret = 0; int server; int peer; char recv_data[DATA_LEN]; struct sockaddr_in server_addr; socklen_t client_addr_len = sizeof(struct sockaddr); server = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP); if (server < 0) { return EXIT_CODE_ERRNO_1; } server_addr.sin_family = AF_INET; server_addr.sin_addr.s_addr = inet_addr(SERVER_IP); server_addr.sin_port = htons(SERVER_PORT); (void)memset_s(&(server_addr.sin_zero), sizeof(server_addr.sin_zero), 0, sizeof(server_addr.sin_zero)); ret = bind(server, const_cast(reinterpret_cast(&server_addr)), sizeof(struct sockaddr)); if (ret != 0) { return EXIT_CODE_ERRNO_2; } ret = listen(server, 1); if (ret < 0) { return EXIT_CODE_ERRNO_3; } peer = accept(server, nullptr, nullptr); if (peer < 0) { return EXIT_CODE_ERRNO_4; } (void)close(server); ret = recv(peer, recv_data, sizeof(recv_data), 0); if (ret < 0) { return EXIT_CODE_ERRNO_5; } ret = strcmp(recv_data, PEER_MSG); if (ret != 0) { return EXIT_CODE_ERRNO_6; } ret = send(peer, SERVER_MSG, strlen(SERVER_MSG) + 1, 0); if (ret < 0) { return EXIT_CODE_ERRNO_7; } (void)close(peer); return 0; } static void *TcpClientThread(void *arg) { int ret = ResetNetaddr(IFNAME, PEER_IP, NETMASK, GW); if (ret == 0) { ret = TcpClient(); } return (void *)(intptr_t)ret; } static int ChildFunc(void *arg) { int ret = ret = ResetNetaddr(IFNAME, SERVER_IP, NETMASK, GW); if (ret != 0) { return EXIT_CODE_ERRNO_1; } return TcpServer(); } VOID ItNetContainer009(void) { int ret = 0; int status; void *tret = nullptr; pthread_t srv; pthread_attr_t attr; char *stack = (char *)mmap(nullptr, STACK_SIZE, PROT_READ | PROT_WRITE, CLONE_STACK_MMAP_FLAG, -1, 0); EXPECT_STRNE(stack, nullptr); char *stackTop = stack + STACK_SIZE; int arg = CHILD_FUNC_ARG; int pid = clone(ChildFunc, stackTop, SIGCHLD | CLONE_NEWNET, &arg); ASSERT_NE(pid, -1); ret = pthread_attr_init(&attr); ASSERT_EQ(ret, 0); ret = pthread_create(&srv, &attr, TcpClientThread, nullptr); ASSERT_EQ(ret, 0); ret = pthread_join(srv, &tret); ASSERT_EQ(ret, 0); ret = pthread_attr_destroy(&attr); ASSERT_EQ(ret, 0); ret = waitpid(pid, &status, 0); ASSERT_EQ(ret, pid); int exitCode = WEXITSTATUS(status); ASSERT_EQ(exitCode, 0); }