diff --git a/CMakeLists.txt b/CMakeLists.txt index 2e1b0d0..2a95b8e 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -38,6 +38,8 @@ set(DRV_SOURCE ${PROJECT_SOURCE_DIR}/drv/drv_do.c # UART驱动源文件 ${PROJECT_SOURCE_DIR}/drv/drv_uart.c + # TCP驱动源文件 + ${PROJECT_SOURCE_DIR}/drv/drv_tcp.c ) # 添加KIT源文件 diff --git a/drv/drv_tcp.c b/drv/drv_tcp.c index e69de29..1c1c6fc 100644 --- a/drv/drv_tcp.c +++ b/drv/drv_tcp.c @@ -0,0 +1,182 @@ +#include "drv_tcp.h" +#define BUFFER_SIZE 1024 + +void getInterfacesAndIPs(char *interfaces[], char *ips[], int *count) { + struct ifconf ifc; + struct ifreq *ifr; + int sockfd; + char buffer[BUFFER_SIZE]; + char ipAddress[INET_ADDRSTRLEN]; + // 创建套接字 + sockfd = socket(AF_INET, SOCK_DGRAM, 0); + if (sockfd < 0) { + perror("创建套接字失败"); + return; + } + ifc.ifc_buf = buffer; + ifc.ifc_len = sizeof(buffer); + // 获取网络接口配置信息 + if (ioctl(sockfd, SIOCGIFCONF, &ifc) < 0) { + perror("获取网络接口配置失败"); + close(sockfd); + return; + } + ifr = (struct ifreq *)ifc.ifc_buf; + int numInterfaces = ifc.ifc_len / sizeof(struct ifreq); + *count = numInterfaces; + for (int i = 0; i < numInterfaces; i++) { + printf("网口名字: %s\n", ifr[i].ifr_name); + interfaces[i] = strdup(ifr[i].ifr_name); + // 获取IP地址 + if (ioctl(sockfd, SIOCGIFADDR, &ifr[i]) < 0) { + perror("获取IP地址失败"); + } else { + struct sockaddr_in *addr = (struct sockaddr_in *)&ifr[i].ifr_addr; + inet_ntop(AF_INET, &addr->sin_addr, ipAddress, INET_ADDRSTRLEN); + printf("IP地址: %s\n", ipAddress); + ips[i] = strdup(ipAddress); + } + } + close(sockfd); +} + + +// 关闭与某个网口相关的套接字并释放内存 +void closeAndFree(int server_fd, int new_socket, char *interface, char *ip) { + close(new_socket); + close(server_fd); + free(interface); + if (ip!= NULL) { + free(ip); + } +} + +// 线程函数,用于处理每个网口的监听和连接 +void *tcpServeThread(void *arg) { + int port = 8888; + int server_fd, new_socket; + struct sockaddr_in address; + int addrlen = sizeof(address); + char buffer[BUFFER_SIZE]; + char *interface = ((char **)arg)[0]; + char *ip = ((char **)arg)[1]; + int optval = 1; + + // printf("Port : %d", port); + // 创建套接字 + label: + if ((server_fd = socket(AF_INET, SOCK_STREAM, 0)) == 0) { + perror("套接字创建失败"); + pthread_exit(NULL); + } + + if (setsockopt(server_fd, SOL_SOCKET, SO_REUSEADDR, &optval, sizeof(optval)) < 0) { + perror("设置套接字选项失败"); + close(server_fd); + exit(EXIT_FAILURE); +} + printf("设置套接字选项成功\n"); + // 设置服务器地址结构体 + address.sin_family = AF_INET; + inet_pton(AF_INET, ip, &address.sin_addr); + address.sin_port = htons(port); + + // 绑定套接字到指定地址和端口 + if (bind(server_fd, (struct sockaddr *)&address, sizeof(address)) < 0) { + perror("绑定失败"); + closeAndFree(server_fd, -1, interface, ip); + pthread_exit(NULL); + } + + // 监听端口,等待客户端连接 + if (listen(server_fd, 3) < 0) { + perror("监听失败"); + closeAndFree(server_fd, -1, interface, ip); + pthread_exit(NULL); + } + + printf("在端口 %s(IP: %s,端口:%d)上启动服务器,正在等待客户端连接...\n", interface, ip, port); + + // 接受客户端连接 + if ((new_socket = accept(server_fd, (struct sockaddr *)&address, (socklen_t *)&addrlen)) < 0) { + perror("接受连接失败"); + closeAndFree(server_fd, -1, interface, ip); + pthread_exit(NULL); + } + + printf("客户端已连接到网口 %s(IP: %s)\n", interface, ip); + + char client_ip[INET_ADDRSTRLEN]; + inet_ntop(AF_INET, &(address.sin_addr), client_ip, INET_ADDRSTRLEN); + printf("Client IP: %s\n", client_ip); + + while (1) { + // 接收客户端发送的数据 + memset(buffer, 0, BUFFER_SIZE); + if (recv(new_socket, buffer, BUFFER_SIZE, 0) < 0) { + perror("接收数据失败"); + break; + } + + if (strlen(buffer) == 0) { + // 客户端断开连接 + printf("客户端已断开连接,从网口 %s(IP: %s)断开。\n", interface, ip); + close(new_socket); + close(server_fd); + sleep(1); + goto label; + } + + printf("端口%s,ip:%s,端口号:%d,收到客户端消息:%s\n", interface, ip, port, buffer); + + // 向客户端发送相同的数据作为回复 + if (send(new_socket, buffer, strlen(buffer), 0) < 0) { + perror("发送数据失败"); + break; + } + } + + // 关闭与该网口相关的套接字并释放内存 + closeAndFree(server_fd, new_socket, interface, ip); + + pthread_exit(NULL); +} + +void tcpServe(int port) { + char *interfaces[100]; + char *ips[100]; + int numInterfaces; + + // 获取本地所有网口名称及对应的IP地址 + getInterfacesAndIPs(interfaces, ips, &numInterfaces); + + pthread_t threads[100]; + char * thread_args[100][3]; + + for (int i = 0; i < numInterfaces; i++) { + if (ips[i] == NULL) { + continue; + } + + // char * portChar; + // sprintf(portChar, "%d", port); + // 为每个线程准备参数 + // thread_args[i][0] = portChar; + thread_args[i][0] = interfaces[i]; + thread_args[i][1] = ips[i]; + + // 创建线程 + if (pthread_create(&threads[i], NULL, tcpServeThread, &thread_args[i])!= 0) { + perror("线程创建失败"); + closeAndFree(-1, -1, interfaces[i], ips[i]); + continue; + } + } + + // 等待所有线程完成 + for (int i = 0; i < numInterfaces; i++) { + if (threads[i]!= 0) { + pthread_join(threads[i], NULL); + } + } +} diff --git a/drv/drv_tcp.h b/drv/drv_tcp.h index e69de29..84c30ab 100644 --- a/drv/drv_tcp.h +++ b/drv/drv_tcp.h @@ -0,0 +1,19 @@ +#ifndef __DRV_TCP_H_ +#define __DRV_TCP_H_ + +#include +#include +#include +#include +#include +#include +#include +#include +#include + +void getInterfacesAndIPs(char *interfaces[], char *ips[], int *numInterfaces); +void closeAndFree(int server_fd, int new_socket, char *interface, char *ip); +void *tcpServeThread(void *arg); +void tcpServe(int port); + +#endif \ No newline at end of file diff --git a/drv/main.c b/drv/main.c new file mode 100644 index 0000000..894a053 --- /dev/null +++ b/drv/main.c @@ -0,0 +1,7 @@ +#include "test.h" + +int main(int argc, char *argv[]) +{ + runTest(); + return 0; +} \ No newline at end of file diff --git a/drv/test.h b/drv/test.h new file mode 100644 index 0000000..b42cd7d --- /dev/null +++ b/drv/test.h @@ -0,0 +1,12 @@ +#ifndef __TEST_H_ +#define __TEST_H_ + +void runTest(); +void testDI(); +void testDO(); +void testUart(); +void testCreatThreadTask(); +void *testLoggerThread(void *arg); +void *testDIDetectThread(void *arg); +void *testUartThread(void *arg); +#endif \ No newline at end of file diff --git a/test/test.c b/test/test.c index a6384f1..592ccc4 100644 --- a/test/test.c +++ b/test/test.c @@ -6,6 +6,7 @@ #include "drv_di.h" #include "drv_do.h" #include "drv_uart.h" +#include "drv_tcp.h" void runTest() { @@ -21,14 +22,16 @@ void testCreatThreadTask() { printf("testThreadTask\n"); logger_init("./log"); - pthread_t tTestLogger, tTestDIDetect, tTestUart; + pthread_t tTestLogger, tTestDIDetect, tTestUart, tTestTcp; // pthread_create(&tTestLogger, NULL, testLoggerThread, "testLoggerThread"); // pthread_join(tTestLogger, NULL); // int ret = pthread_create(&tTestDIDetect, NULL, testDIDetectThread, "testDIDetectThread"); // printf("testDIDetectThread ret = %d\n", ret); // pthread_join(tTestDIDetect, NULL); - pthread_create(&tTestUart, NULL, testUartThread, "testUartThread"); - pthread_join(tTestUart, NULL); + // pthread_create(&tTestUart, NULL, testUartThread, "testUartThread"); + // pthread_join(tTestUart, NULL); + pthread_create(&tTestTcp, NULL, testTcpThread, "testTcpThread"); + pthread_join(tTestTcp, NULL); logger_destroy(); } @@ -112,4 +115,13 @@ void testUart() char test[100]="forlinx_uart_test.1234567890..."; printf("Send test data:\n%s\n", test); write(devFd, test, strlen(test) + 1); -} \ No newline at end of file +} + +void *testTcpThread(void *arg) +{ + logger_level_printf(LOGGER_DEBUG_LEVEL, arg); + printf("TCP Serve Start\n"); + // getInterfacesAndIPs(); + tcpServe(8888); + +}