explain
1. Reference code
https://www.cnblogs.com/orlion/p/6119812.html
#include <string.h> #include <stdio.h> #include <stdlib.h> #include <netinet/in.h> #include <arpa/inet.h> #include <ctype.h> #include "wrap.h" #define MAXLINE 80 #define SERV_PORT 8001 int main(int argc, char **argv) { int i, maxi, maxfd, listenfd, connfd, sockfd; int nready, client[FD_SETSIZE]; ssize_t n; fd_set rset, allset; char buf[MAXLINE]; char str[INET_ADDRSTRLEN]; socklen_t cliaddr_len; struct sockaddr_in cliaddr, servaddr; listenfd = Socket(AF_INET, SOCK_STREAM, 0); bzero(&servaddr, sizeof(servaddr)); servaddr.sin_family = AF_INET; servaddr.sin_addr.s_addr = htonl(INADDR_ANY); servaddr.sin_port = htons(SERV_PORT); Bind(listenfd, (struct sockaddr *)&servaddr, sizeof(servaddr)); Listen(listenfd, 20); maxfd = listenfd; maxi = -1; for (i = 0; i < FD_SETSIZE; i++) client[i] = -1; /* -1 indicates available entry */ FD_ZERO(&allset); FD_SET(listenfd, &allset); for ( ; ; ) { rset = allset; /* structure assignment */ nready = select(maxfd+1, &rset, NULL, NULL, NULL); if (nready < 0) perr_exit("select error"); if (FD_ISSET(listenfd, &rset)) { /* new client connection */ cliaddr_len = sizeof(cliaddr); connfd = Accept(listenfd, (struct sockaddr *)&cliaddr, &cliaddr_len); printf("received from %s at PORT %d\n", inet_ntop(AF_INET, &cliaddr.sin_addr, str, sizeof(str)), ntohs(cliaddr.sin_port)); for (i = 0; i < FD_SETSIZE; i++) if (client[i] < 0) { client[i] = connfd; /* save descriptor */ break; } if (i == FD_SETSIZE) { fputs("too many clients\n", stderr); exit(1); } FD_SET(connfd, &allset); /* add new descriptor to set */ if (connfd > maxfd) maxfd = connfd; /* for select */ if (i > maxi) maxi = i; /* max index in client[] array */ if (--nready == 0) continue; /* no more readable descriptors */ } for (i = 0; i <= maxi; i++) { /* check all clients 714 for data */ if ( (sockfd = client[i]) < 0) continue; if (FD_ISSET(sockfd, &rset)) { if ( (n = Read(sockfd, buf, MAXLINE)) == 0) { Close(sockfd); FD_CLR(sockfd, &allset); client[i] = -1; } else { int j; for (j = 0; j < n; j++) buf[j] = toupper(buf[j]); Write(sockfd, buf, n); } if (--nready == 0) break; /* no more readable descriptors */ } } } }
#include <stdlib.h> #include <stdio.h> #include <errno.h> #include <sys/socket.h> #include <netinet/in.h> #include <string.h> #include <unistd.h> void perr_exit(const char *s) { perror(s); exit(1); } int Accept(int fd, struct sockaddr *sa, socklen_t *salenptr) { int n; again: if ((n = accept(fd, sa, salenptr)) < 0) { if ((errno == ECONNABORTED) || (errno == EINTR)) goto again; else perr_exit("accept error"); } return n; } void Bind(int fd, struct sockaddr *sa, socklen_t salen) { if (bind(fd, sa, salen) < 0) perr_exit("bind error"); } void Connect(int fd, const struct sockaddr *sa, socklen_t salen) { if (connect(fd, sa, salen) < 0) perr_exit("connent error"); } void Listen(int fd, int backlog) { if (listen(fd, backlog) < 0) perr_exit("listen error"); } int Socket(int family, int type, int protocol) { int n; if ((n = socket(family, type, protocol)) < 0) perr_exit("socket error"); return n; } ssize_t Read(int fd, void *ptr, size_t nbytes) { ssize_t n; again: if ((n = read(fd, ptr, nbytes)) < 0) { if (errno == EINTR) goto again; else return -1; } return n; } ssize_t Write(int fd, const void *ptr, size_t nbytes) { ssize_t n; again: if ((n = write(fd, ptr, nbytes)) == -1) { if (errno == EINTR) goto again; else return -1; } return n; } void Close(int fd) { if (close(fd) == -1) perr_exit("close error"); } ssize_t Readn(int fd, void *vptr, size_t n) { size_t nleft; ssize_t nread; char *ptr; ptr = vptr; nleft = n; while (nleft > 0) { if ((nread = read(fd, ptr, nleft)) < 0) { if (errno == EINTR) nread = 0; else return -1; } else if (nread == 0) { break; } nleft -= nread; ptr += nread; } return n - nleft; } ssize_t Writen(int fd, const void *vptr, size_t n) { size_t nleft; ssize_t nwritten; const char *ptr; ptr = vptr; nleft = n; while (nleft > 0) { if ((nwritten = write(fd, ptr, nleft)) <= 0) { if (nwritten < 0 && errno == EINTR) nwritten = 0; else return -1; } nleft -= nwritten; ptr += nwritten; } return n; } static ssize_t my_read(int fd, char *ptr) { static int read_cnt; static char *read_ptr; static char read_buf[100]; if (read_cnt <= 0) { again: if ((read_cnt = read(fd, read_buf, sizeof(read_buf))) < 0) { if (errno == EINTR) goto again; return -1; } else if (read_cnt == 0) return 0; read_ptr = read_buf; } read_cnt--; *ptr = *read_ptr++; return 1; } ssize_t Readline(int fd, void *vptr, size_t maxlen) { ssize_t n, rc; char c, *ptr; ptr = vptr; for (n = 1; n < maxlen; n++) { if ((rc = my_read(fd, &c)) == 1) { *ptr++ = c; if (c == '\n') break; } else if (rc == 0) { *ptr = 0; return n - 1; } else { return -1; } } *ptr = 0; return n; }
2. Description
Because the current project needs to achieve high-speed and efficient transmission, a set of TCP server program is encapsulated by lwip's select
It is also recommended that big guys use this program as a TCP server. In this way, they can use this bottom layer for similar projects in the future
Download program to development board
1. Put the code of this section in the English directory
data:image/s3,"s3://crabby-images/cf909/cf909e6aed6b3762094a35d2d29277ad819cf9fb" alt=""
2. Right click and select open with VScode
data:image/s3,"s3://crabby-images/eb09a/eb09ad367e4d3ec4cfe2542a1a12d5d1ea021a16" alt=""
3. About partial configuration
The user can configure the module hotspot name and the router information connected to the module in this function file
data:image/s3,"s3://crabby-images/9590d/9590d6731483656485ca4dbfdfcac1feca686bbf" alt=""
If you don't need to connect the router or modify it, you can't connect internally at most
data:image/s3,"s3://crabby-images/033c5/033c56c7a3296b6a1295ad802f9beafde798a837" alt=""
The user can set the port number of the TCP server listening here: 8888 is listening now
data:image/s3,"s3://crabby-images/2556f/2556f2b7a78109789c75d5abb7531785545c5ecd" alt=""
4. Compile and download to the development board (the first compilation takes a long time)
data:image/s3,"s3://crabby-images/00e28/00e288090fd6aac2f6fd76e105e772e92480e6ea" alt=""
test
1. After the program is downloaded, there will be a program named ESP32_WIFI hotspots
data:image/s3,"s3://crabby-images/43cf4/43cf443c537e2ebf3d29e9ecd1e864fa4c1ff2c3" alt=""
2. If the module is connected to the router, the log will also print the information after connecting to the router
data:image/s3,"s3://crabby-images/bc047/bc0472a2d09ffcc45bfdaf0513a50a4edc17a113" alt=""
3. Prompt
If the big guy uses the hotspot of the mobile phone or computer connection module to test,
Then the IP address of the TCP server of the module is 192.168 4.1 port number: 8888
Now my computer and module are in a router. Ha, I use 192.168 0.102 address test
4. Open the network debugging assistant test
What is written in the program is to return whatever data is received
data:image/s3,"s3://crabby-images/f3f95/f3f955087bd0090bd7a91aab72e7deb9f6ad5543" alt=""
Add another client
data:image/s3,"s3://crabby-images/30cec/30cec46fa3612f175e90fe2bff717a424da7e12d" alt=""
Program instructions (how to use it first)
1. If users need to migrate, they can directly put the following files into their own project
data:image/s3,"s3://crabby-images/6e6ce/6e6ced4510de788fdec913fc110ba2ded33374b6" alt=""
2. Create a TCP server (see the following figure for each parameter)
data:image/s3,"s3://crabby-images/85260/85260a603140222a5499794f2d56bd75d99a3b16" alt=""
data:image/s3,"s3://crabby-images/29227/29227f2add7ef0ec03fdcf8f2b90c66a696f7306" alt=""
3. The data received by the server is in this function (this function is in the TCP listening task. Be careful not to block it in this function)
data:image/s3,"s3://crabby-images/324f0/324f0ca5414967f721b1577c7157238d89427f04" alt=""
4. About sending data to the client
1. There are two TCP functions for sending data to the client_ server_ select_ Write and tcp_server_select_send
data:image/s3,"s3://crabby-images/9158d/9158ddd142a2d516eefd3851ca96e272d9766fe4" alt=""
2, tcp_server_select_write means that it can only be called in the received data
3. Suppose you need to send the data received from the serial port to all TCP clients_ server_ select_ Send (- 1, data address, data length)
data:image/s3,"s3://crabby-images/118b1/118b140325eed871922c229e0c51a7ab3b1e33f0" alt=""
4. Suppose you need to send the data received from the serial port to the specified TCP client, you need to get the client's index in the receiving function first
I'm just giving an example. Generally, after receiving what data, I assign the value to which client the following data is sent
data:image/s3,"s3://crabby-images/ac55f/ac55f8bbfb8eead8b03931f2772e624fa2aa5319" alt=""
data:image/s3,"s3://crabby-images/3ec0e/3ec0e21fc958b0c5051ddf3134ea7cb0455c682a" alt=""
Program description
1. Create a TCP server
data:image/s3,"s3://crabby-images/7d8f7/7d8f786cc6f8393249a12557c87b267c92d7cd2e" alt=""
data:image/s3,"s3://crabby-images/5eda3/5eda3b2d798249117a40b30fa36a44abce7be026" alt=""
2. The TCP server monitors the connection and receives data
data:image/s3,"s3://crabby-images/de86f/de86f802ff3be9a9b340abd389dd93ccb0f7b5bf" alt=""
data:image/s3,"s3://crabby-images/8d55a/8d55af8812c290a73134ba71b2667a3110fde2ae" alt=""
data:image/s3,"s3://crabby-images/2af72/2af729083de6c22b3413905699be4e83970933a6" alt=""
data:image/s3,"s3://crabby-images/8cc71/8cc715129471b353be39442781b9445be9c709ca" alt=""
data:image/s3,"s3://crabby-images/547b4/547b49d3b8a07d95eac5f73892e897b79b4f449c" alt=""
3. Send data
Ringbuffer + semaphore + task is used to send data
data:image/s3,"s3://crabby-images/4770e/4770edd46d71b7be43a01559395a1645ee0c79c5" alt=""
When sending data, the number is stored in Ringbuffer and the semaphore is + 1
data:image/s3,"s3://crabby-images/b61bf/b61bf89e725bf0ea23580aff6f39b3c3ff829146" alt=""
Get the semaphore in the task, then get the data in the cache, and then send the data to the client
data:image/s3,"s3://crabby-images/17de1/17de1a48d350886712895b561dd09aaf63ddf952" alt=""