/* * 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 = 8000; static const char *PEER_IP = "192.168.100.5"; static const int PEER_PORT = 8001; static const int DATA_LEN = 128; 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 int UdpClient(void) { int ret = 0; int peer; int try_count = TRY_COUNT; char recv_data[DATA_LEN]; struct sockaddr_in server_addr; struct sockaddr_in peer_addr; peer = socket(AF_INET, SOCK_DGRAM, 0); 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)); peer_addr.sin_family = AF_INET; peer_addr.sin_addr.s_addr = inet_addr(PEER_IP); peer_addr.sin_port = htons(PEER_PORT); (void)memset_s(&(peer_addr.sin_zero), sizeof(peer_addr.sin_zero), 0, sizeof(peer_addr.sin_zero)); ret = bind(peer, const_cast(reinterpret_cast(&peer_addr)), sizeof(struct sockaddr)); if (ret != 0) { return EXIT_CODE_ERRNO_2; } timeval tv = {1, 0}; ret = setsockopt(peer, SOL_SOCKET, SO_RCVTIMEO, const_cast(reinterpret_cast(&tv)), sizeof(timeval)); /* loop try util server is ready */ while (try_count--) { ret = sendto(peer, PEER_MSG, strlen(PEER_MSG) + 1, 0, const_cast(reinterpret_cast(&server_addr)), (socklen_t)sizeof(server_addr)); if (ret == -1) { continue; } ret = recvfrom(peer, recv_data, DATA_LEN, 0, nullptr, nullptr); if (ret != -1) { break; } } if (ret < 0) { return EXIT_CODE_ERRNO_3; } (void)close(peer); ret = strcmp(recv_data, SERVER_MSG); if (ret != 0) { return EXIT_CODE_ERRNO_4; } return 0; } static int ChildFunc(void *arg) { int ret = NetContainerResetNetAddr(IFNAME, PEER_IP, NETMASK, GW); if (ret != 0) { return EXIT_CODE_ERRNO_1; } return UdpClient(); } static int UdpServer(void) { int ret = 0; int server; int try_count = TRY_COUNT; char recv_data[DATA_LEN]; struct sockaddr_in server_addr; struct sockaddr_in peer_addr; socklen_t peer_addr_len = sizeof(struct sockaddr); server = socket(AF_INET, SOCK_DGRAM, 0); 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 = recvfrom(server, recv_data, DATA_LEN, 0, reinterpret_cast(&peer_addr), &peer_addr_len); if (ret < 0) { return EXIT_CODE_ERRNO_3; } ret = sendto(server, SERVER_MSG, strlen(SERVER_MSG) + 1, 0, const_cast(reinterpret_cast(&peer_addr)), peer_addr_len); if (ret < 0) { return EXIT_CODE_ERRNO_4; } (void)close(server); ret = strcmp(recv_data, PEER_MSG); if (ret != 0) { return EXIT_CODE_ERRNO_5; } return ret; } static void *UdpServerThread(void *arg) { int ret = NetContainerResetNetAddr(IFNAME, SERVER_IP, NETMASK, GW); if (ret != 0) { return (void *)(intptr_t)ret; } ret = UdpServer(); return (void *)(intptr_t)ret; } void ItNetContainer005(void) { int ret = 0; int status; void *tret = nullptr; pthread_t srv; pthread_attr_t attr; ret = pthread_attr_init(&attr); ASSERT_EQ(ret, 0); ret = pthread_create(&srv, &attr, UdpServerThread, nullptr); ASSERT_EQ(ret, 0); 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; auto pid = clone(ChildFunc, stackTop, SIGCHLD | CLONE_NEWNET, &arg); ASSERT_NE(pid, -1); ret = waitpid(pid, &status, 0); ASSERT_EQ(ret, pid); int exitCode = WEXITSTATUS(status); ASSERT_EQ(exitCode, 0); ret = pthread_join(srv, &tret); ASSERT_EQ(ret, 0); ret = pthread_attr_destroy(&attr); ASSERT_EQ(ret, 0); }