IPv6 Socket API实例
IPv4/v6 Socket编程异同
在基于Socket API进行开发的应用程序中,IPv4/v6都使用基本相同的编程模型;如connect、accept、listen、send/sendto、recv/recvfrom等Socket API函数在IPv4/v6中用法也基本一致,而IPv4与IPv6间的主要差异体现在了与地址相关的API函数上,其具体差异如下图所示:
getaddrinfo()函数在lwIP中实现不完善,这里只介绍inet_pton()函数的用法:
函数原型:inet_pton(int family, const char _strptr, void _addrptr)
返回值:若成功返回1,若输入的不是有效表达式返回0,若出错则返回-1
参数:family:可以是AF_INET,也可以是AF_INET6,如果以不被支持的地址族作为参数,函数将返回错误,并置errno为EAFNOSUPPORT
作用:该函数尝试转换由strptr指向的字符串,并通过addrptr指针存放二进制结果,完成了表达式到IPv6地址的转换。
关于getaddrinfo()、inet_ntop()函数的用法请参见Unix网络编程卷一。
PC测试程序
往往对IPv4的TCP/UDP的SERVer/Client进行测试时,Windows下有许多网络调试工具,而IPv6的测试主要是在Linux下利用Socket API编写的测试程序,具体请见GitHub,请在Linux的命令行下完成测试。
TCP Server例子
下面是一个在RT-Thread上使用BSD socket接口的TCP服务端例子。当把这个代码加入到RT-Thread时,它会自动向finsh命令行添加一个tcpserver6命令,在finsh上执行tcpserver6()函数即可启动这个TCP服务端,该TCP服务端在端口10001上进行监听。
当有TCP客户向这个服务端进行连接后,只要服务端接收到数据,它就会立即向客户端发送“This is TCP Server from RT-Thread.”的字符串。
如果服务端接收到q或Q字符串时,服务器将主动关闭这个TCP连接。如果服务端接收到exit字符串时,那么将退出服务。
- #include <rtthread.h>
- #include <lwip/sockets.h> /* 使用BSD Socket需包含socket.h */
- #define SERV_PORT 10001 /* 服务器使用的端口号 */
- #define BUF_SIZE 1024 /* 接收缓冲区的长度 */
- #define BACKLOG 5 /* 请求队列的长度 */
- static const char send_data[] = "This is TCP Server from RT-Thread.";
- void tcpserver6(void)
- {
- int sockfd, clientfd;
- struct sockaddr_in6 server_addr6, client_addr6;
- int bytes_received;
- char *recv_data;
- rt_uint32_t sin_size;
- rt_bool_t stop = RT_FALSE;
- /* 分配接收用的数据缓冲 */
- recv_data = rt_malloc(BUF_SIZE);
- if(recv_data == RT_NULL)
- {
- /* 分配内存失败,返回 */
- rt_kprintf("No memory\n");
- return ;
- }
- /* 创建一个socket,family类型为PF_INET6,套接字类型为SOCK_STREAM */
- if((sockfd = socket(PF_INET6, SOCK_STREAM, 0)) == -1)
- {
- rt_kprintf("Socket error\n");
- rt_free(recv_data);
- return ;
- }
- /* 初始化服务端地址 */
- server_addr6.sin6_family = AF_INET6;
- memcpy(server_addr6.sin6_addr.s6_addr, IP6_ADDR_ANY, 16);
- server_addr6.sin6_port = htons(SERV_PORT);
- /* 绑定Socket到服务端地址 */
- if(bind(sockfd, (struct sockaddr *)&server_addr6, sizeof(struct sockaddr)) == -1)
- {
- rt_kprintf("Bind error\n");
- rt_free(recv_data);
- return ;
- }
- /* 在Socket上进行监听 */
- if(listen(sockfd, BACKLOG) == -1)
- {
- rt_kprintf("Listen error\n");
- rt_free(recv_data);
- return ;
- }
- rt_sprintf(recv_data, "%4d", SERV_PORT);
- rt_kprintf("\nTCPServer Waiting for client on port %s...\n", recv_data);
- while(stop != RT_TRUE)
- {
- sin_size = sizeof(struct sockaddr_in6);
- /*接收客户端的连接*/
- clientfd = accept(sockfd, (struct sockaddr *)&client_addr6, &sin_size);
- rt_kprintf("I got a connection from (IP:%s, PORT:%d\n)", inet6_ntoa(client_addr6.sin6_addr), ntohs(client_addr6.sin6_port));
- /* 客户端连接的处理 */
- while(1)
- {
- /* 发送数据到connected socket */
- send(clientfd, send_data, strlen(send_data), 0);
- /* 接收数据,但并不一定能够收到BUF_SIZE大小的数据 */
- bytes_received = recv(clientfd, recv_data, BUF_SIZE, 0);
- if(bytes_received <= 0)
- {
- /* 接收失败,关闭这个Connected Socket */
- closesocket(clientfd);
- break;
- }
- /* 收到数据,加入字符串结束符*/
- recv_data[bytes_received] = '\0';
- if(strcmp(recv_data, "q") == 0 || strcmp(recv_data, "Q") == 0)
- {
- /* 关闭连接 */
- closesocket(clientfd);
- break;
- }
- else if(strcmp(recv_data, "exit") == 0)
- {
- /* 关闭服务端 */
- closesocket(clientfd);
- stop = RT_TRUE;
- break;
- }
- else
- {
- /* 打印收到的数据 */
- rt_kprintf("RECEIVED DATA = %s\n", recv_data);
- }
- }
- }
- closesocket(sockfd);
- rt_free(recv_data);
- return ;
- }
- #ifdef RT_USING_FINSH
- #include <finsh.h>
- FINSH_FUNCTION_EXPORT(tcpserver6, start tcp server via ipv6 );
- #endif
TCP Client例子
下面则是另一个如在RT-Thread上使用BSD socket接口的TCP客户端例子。当把这个代码加入到RT-Thread时,它会自动向finsh 命令行添加一个tcpclient6命令,在finsh上执行tcpclient6()函数即可启动这个TCP服务端。
当TCP客户端连接成功时,它会接收服务端传过来的数据。当有数据接收到时,如果是以q或Q开头,它将主动断开这个连接;否则会把接收的数据在控制终端中打印出来,然后发送“This is TCP Client from RT-Thread.”的字符串。
- #include <rtthread.h>
- #include <lwip/netdb.h> /*为解析主机名,需包含netdb.h */
- #include <lwip/sockets.h>
- #define SERV_PORT 12345 /* 服务器端口号 */
- #define SERVADDR "4006:e024:680:c6e:223:8bff:fe59:de90" /* 由于未实现Scope id,请不要使用link-local Address */
- #define BUF_SIZE 1024 /* 缓冲区的长度 */
- static const char send_data[] = "This is TCP Client from RT-Thread.";
- void tcpclient6(void)
- {
- char* recv_data;
- int sockfd, bytes_received;
- struct sockaddr_in6 server_addr6;
- int status = 0;
- /* 分配接收用的数据缓冲 */
- recv_data = rt_malloc(BUF_SIZE);
- if(recv_data == RT_NULL)
- {
- /* 分配内存失败,返回 */
- rt_kprintf("No memory\n");
- return ;
- }
- /* 创建一个socket,family类型为PF_INET6,套接字类型为SOCK_STREAM */
- if((sockfd = socket(PF_INET6, SOCK_STREAM, 0)) == -1)
- {
- rt_kprintf("Socket error\n");
- rt_free(recv_data);
- return ;
- }
- /* 初始化预连接的服务端地址 */
- memset(&server_addr6, 0, sizeof(server_addr6));
- server_addr6.sin6_family = AF_INET6;
- server_addr6.sin6_port = htons(SERV_PORT);
- /* 将字符串转换为IPv6地址 */
- if(inet_pton(AF_INET6, SERVADDR, &server_addr6.sin6_addr.s6_addr) != 1)
- {
- rt_kprintf("inet_pton() error\n");
- rt_free(recv_data);
- return ;
- }
- /* 连接到服务端 */
- status = connect(sockfd, (struct sockaddr *)&server_addr6, sizeof(server_addr6));
- if(status < 0)
- {
- /* 连接失败 */
- rt_kprintf("Connect error:%d\n", status);
- rt_free(recv_data);
- return ;
- }
- while(1)
- {
- /* 从socket连接中接收最大BUF_SIZE字节数据 */
- bytes_received = recv(sockfd, recv_data, BUF_SIZE -1, 0);
- if(bytes_received <= 0)
- {
- /* 接收失败,关闭这个连接 */
- closesocket(sockfd);
- rt_free(recv_data);
- break;
- }
- /* 收到数据,加入字符串结束符*/
- recv_data[bytes_received] = '\0';
- if(strcmp(recv_data, "q") == 0 || strcmp(recv_data, "Q") == 0)
- {
- /* 关闭这个连接 */
- closesocket(sockfd);
- rt_free(recv_data);
- break;
- }
- else
- {
- /* 打印收到的数据 */
- rt_kprintf("\nReceived data = %s ", recv_data);
- }
- /* 发送数据到服务端 */
- send(sockfd, send_data, strlen(send_data), 0);
- }
- return;
- }
- #ifdef RT_USING_FINSH
- #include <finsh.h>
- FINSH_FUNCTION_EXPORT(tcpclient6, startup tcp client via ipv6);
- #endif
UDP Server例子
下面是一个在RT-Thread上使用BSD socket接口的UDP服务端例子,当把这个代码加入到RT-Thread操作系统时,它会自动向finsh命令行添加一个udpserver6命令,在finsh上执行udpserver6()函数即可启动这个UDP服务端,该UDP服务端在端口10012上进行监听。
当服务端接收到数据时,它将把数据打印到控制终端中;如果服务端接收到exit字符串时,那么服务端将退出服务。
- #include <rtthread.h>
- #include <lwip/sockets.h>
- #define SERV_PORT 10012
- #define BUF_SIZE 1024
- static const char send_data[] = "This is UDP Server from RT-Thread.";
- void udpserver6(void)
- {
- int sockfd;
- struct sockaddr_in6 server_addr6;
- struct sockaddr_in6 client_addr6;
- int bytes_read;
- char *recv_data;
- rt_uint32_t addr_len;
- /* 分配接收用的数据缓冲 */
- recv_data = rt_malloc(BUF_SIZE);
- if(recv_data == RT_NULL)
- {
- /* 分配内存失败,返回 */
- rt_kprintf("No memory\n");
- return ;
- }
- /* 创建一个socket,family类型为PF_INET6,套接字类型为SOCK_DGRAM */
- if((sockfd = socket(AF_INET6, SOCK_DGRAM, 0)) == -1)
- {
- rt_kprintf("Socket error\n");
- rt_free(recv_data);
- return ;
- }
- /* 初始化服务端地址 */
- server_addr6.sin6_family = AF_INET6;
- server_addr6.sin6_port = htons(SERV_PORT);
- memcpy(server_addr6.sin6_addr.s6_addr, IP6_ADDR_ANY, 16);
- /* 绑定Socket到服务端地址 */
- if(bind(sockfd, (struct sockaddr *)&server_addr6, sizeof(struct sockaddr)) == -1)
- {
- /* 绑定地址失败 */
- rt_kprintf("Bind error\n");
- rt_free(recv_data);
- return ;
- }
- rt_sprintf(recv_data, "%4d", SERV_PORT);
- rt_kprintf("UDPServer Waiting for client on port %s...\n", recv_data);
- addr_len = sizeof(struct sockaddr);
- while(1)
- {
- /* 从socket中收取最大BUF_SIZE字节数据 */
- bytes_read = recvfrom(sockfd, recv_data, BUF_SIZE - 1, 0, (struct sockaddr *)&client_addr6, &addr_len);
- /* 收到数据,加入字符串结束符*/
- recv_data[bytes_read] = '\0';
- /* 输出接收的数据 */
- rt_kprintf("\n(%s, %d) said:", inet6_ntoa(client_addr6.sin6_addr), ntohs(client_addr6.sin6_port));
- rt_kprintf("%s", recv_data);
- if(strcmp(recv_data, "exit") == 0)
- {
- /* 关闭服务端 */
- closesocket(sockfd);
- rt_free(recv_data);
- break;
- }
- }
- return ;
- }
- #ifdef RT_USING_FINSH
- #include <finsh.h>
- FINSH_FUNCTION_EXPORT(udpserver6, startup udp server via ipv6);
- #endif
UDP Client例子
下面是另一个在RT-Thread上使用BSD socket接口的UDP客户端例子。当把这个代码加入到RT-Thread时,它会自动向finsh命令行添加一个udpclient6命令,在finsh上执行udpclient6()函数即可启动这个UDP客户端。当UDP客户端启动后,它将发送“This is UDP Client from RT-Thread.”的字符串给服务端,然后接收数据并打印数据。
- #include <rtthread.h>
- #include <lwip/netdb.h> /*为解析主机名,需包含netdb.h */
- #include <lwip/sockets.h>
- #define SERV_PORT 22345 /* 服务器端口号 */
- #define SERVADDR "4006:e024:680:c6e:223:8bff:fe59:de90" /* 由于未实现Scope id,请不要使用link-local Address */
- #define BUF_SIZE 1024 /* 缓冲区的长度 */
- static const char send_data[] = "This is UDP Client from RT-Thread.";
- void udpclient6(void)
- {
- char *recv_data;
- int sockfd;
- struct sockaddr_in6 server_addr6, client_addr6;
- socklen_t clientlen;
- /* 分配接收用的数据缓冲 */
- recv_data = rt_malloc(BUF_SIZE);
- if(recv_data == RT_NULL)
- {
- rt_kprintf("No memory\n");
- return ;
- }
- /* 创建一个socket,family类型为PF_INET6,套接字类型为SOCK_DGRAM */
- if((sockfd = socket(PF_INET6, SOCK_DGRAM, 0)) == -1)
- {
- rt_kprintf("Socket error\n");
- rt_free(recv_data);
- return ;
- }
- /* 初始化预连接的服务端地址 */
- memset(&server_addr6, 0, sizeof(server_addr6));
- server_addr6.sin6_family = AF_INET6;
- server_addr6.sin6_port = htons(SERV_PORT);
- /* 将字符串转换为IPv6地址 */
- if(inet_pton(AF_INET6, SERVADDR, &server_addr6.sin6_addr.s6_addr) != 1)
- {
- rt_kprintf("inet_pton() error\n");
- rt_free(recv_data);
- return ;
- }
- /* 发送数据到服务端 */
- if(sendto(sockfd, send_data, sizeof(recv_data), 0, (struct sockaddr *)&server_addr6, sizeof(server_addr6)) < 0)
- {
- rt_kprintf("Sendto error\n");
- rt_free(recv_data);
- return ;
- }
- rt_kprintf("Waiting for a reply...\n");
- clientlen = sizeof(client_addr6);
- /* 接收数据 */
- if(recvfrom(sockfd, recv_data, BUF_SIZE, 0, (struct sockaddr *)&client_addr6, &clientlen) < 0)
- {
- /* 接收失败 */
- rt_kprintf("Recvfrom error\n");
- rt_free(recv_data);
- return ;
- }
- /* 打印数据 */
- rt_kprintf("got '%s'\n", recv_data);
- closesocket(sockfd);
- }
- #ifdef RT_USING_FINSH
- #include <finsh.h>
- FINSH_FUNCTION_EXPORT(udpclient6, start udp server via ipv6);
- #endif