LWIP开发 | 通过scoket实现tcp server

LWIP为了提高兼容性也可以通过Scoket实现,主要是代码函数在文件#include <lwip/sockets.h>中,以下是主要相关代码

/** @ingroup socket */
#define accept(s,addr,addrlen)                    lwip_accept(s,addr,addrlen)
/** @ingroup socket */
#define bind(s,name,namelen)                      lwip_bind(s,name,namelen)
/** @ingroup socket */
#define shutdown(s,how)                           lwip_shutdown(s,how)
/** @ingroup socket */
#define getpeername(s,name,namelen)               lwip_getpeername(s,name,namelen)
/** @ingroup socket */
#define getsockname(s,name,namelen)               lwip_getsockname(s,name,namelen)
/** @ingroup socket */
#define setsockopt(s,level,optname,opval,optlen)  lwip_setsockopt(s,level,optname,opval,optlen)
/** @ingroup socket */
#define getsockopt(s,level,optname,opval,optlen)  lwip_getsockopt(s,level,optname,opval,optlen)
/** @ingroup socket */
#define closesocket(s)                            lwip_close(s)
/** @ingroup socket */
#define connect(s,name,namelen)                   lwip_connect(s,name,namelen)
/** @ingroup socket */
#define listen(s,backlog)                         lwip_listen(s,backlog)
/** @ingroup socket */
#define recv(s,mem,len,flags)                     lwip_recv(s,mem,len,flags)
/** @ingroup socket */
#define recvfrom(s,mem,len,flags,from,fromlen)    lwip_recvfrom(s,mem,len,flags,from,fromlen)
/** @ingroup socket */
#define send(s,dataptr,size,flags)                lwip_send(s,dataptr,size,flags)
/** @ingroup socket */
#define sendmsg(s,message,flags)                  lwip_sendmsg(s,message,flags)
/** @ingroup socket */
#define sendto(s,dataptr,size,flags,to,tolen)     lwip_sendto(s,dataptr,size,flags,to,tolen)
/** @ingroup socket */
#define socket(domain,type,protocol)              lwip_socket(domain,type,protocol)
/** @ingroup socket */
#define select(maxfdp1,readset,writeset,exceptset,timeout)     lwip_select(maxfdp1,readset,writeset,exceptset,timeout)
/** @ingroup socket */
#define ioctlsocket(s,cmd,argp)                   lwip_ioctl(s,cmd,argp)

TCP程序编写
实现环境
1.移植Freertos
2.移植LWIP,开启LWIP_SOCKET
在这里插入图片描述

#include "tcp_server_test.h"
#include "lwip/opt.h"
#include <lwip/sockets.h>
#include "lwip/sys.h"
#include "lwip/api.h"
#include "string.h"

#if LWIP_SOCKET	//需要开启Scoket才能使用

#define RECV_DATA         	(1024UL)
#define LOCAL_PORT 			(6133UL)
#define BACKLOG 			(5UL)/*最大监听数*/ 

#define LWIP_TCP_DEBUG_ENABLE    1

#if LWIP_TCP_DEBUG_ENABLE
   #define LWIP_TCP_DEBUG	printf
#else
  #define LWIP_TCP_DEBUG(...)
#endif


static void tcp_server_thread(void *arg)
{
  int sockfd = -1,connected; /*socket句柄和建立连接后的句柄*/
  struct sockaddr_in server_addr,client_addr;
  socklen_t sin_size;
  int recv_data_len;
  uint8_t recv_data[RECV_DATA];
	
  LWIP_TCP_DEBUG("tcp server port %d\n\n",LOCAL_PORT);
	
again: 
	
  //创建scoket描述符 AF_INET 使用ipv4地址
  sockfd = socket(AF_INET, SOCK_STREAM, 0);  
  if (sockfd < 0)
  {
      LWIP_TCP_DEBUG("Socket error\n");
	  close(sockfd);
	  vTaskDelay(100);
      goto again;
  }
  //
  server_addr.sin_family = AF_INET; 				//该属性表示接收本机或其他机器传输
  server_addr.sin_addr.s_addr = htonl(INADDR_ANY);	//本机IP
  server_addr.sin_port = htons(LOCAL_PORT);			//端口号
  memset(&(server_addr.sin_zero), 0, sizeof(server_addr.sin_zero));
  
  if (bind(sockfd, (struct sockaddr *)&server_addr, sizeof(struct sockaddr)) == -1)
  {
      LWIP_TCP_DEBUG("Unable to bind\n");
	  close(sockfd);
	  vTaskDelay(100);
      goto again;
  }
  
  //设置最大监听数
  if (listen(sockfd, BACKLOG) == -1)
  {
      LWIP_TCP_DEBUG("Listen error\n");
	  close(sockfd);
	  vTaskDelay(100);
      goto again;
  }
  
  while(1)
  {
    sin_size = sizeof(struct sockaddr_in);

	//在这里阻塞知道接收到消息,参数分别是socket句柄,接收到的地址信息以及大小 
    connected = accept(sockfd, (struct sockaddr *)&client_addr, &sin_size);

    LWIP_TCP_DEBUG("new client connected from (%s, %d)\n", inet_ntoa(client_addr.sin_addr), ntohs(client_addr.sin_port));
    
    int tcp_nodelay = 1;//don't delay send to coalesce packets
    setsockopt(connected,IPPROTO_TCP,TCP_NODELAY,(void *) &tcp_nodelay,sizeof(int));    			
    
    while(1)
    {
      recv_data_len = recv(connected, recv_data, RECV_DATA, 0);
      
      if (recv_data_len <= 0)
	  {
		   break;
	  }		  
        
      
      LWIP_TCP_DEBUG("recv %d len data\n",recv_data_len);
	  //发送内容
      write(connected,recv_data,recv_data_len); 
    }
	
    if (connected >= 0)
	{
		 close(connected);
	}
	
    connected = -1;  
  }
  
  if (sockfd >= 0)
  {
	 close(sockfd);
  }	  
	  
}
#endif

void tcp_server_init(void)
{
#if LWIP_SOCKET
  sys_thread_new("tcpecho_thread", tcp_server_thread, NULL, 512, 4);
#endif
}

验证如下
在这里插入图片描述
推荐一个很好的博主:https://blog.csdn.net/Chuangke_Andy/article/details/121420376

  • 2
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
实现STM32 FreeRTOS LwIP TCP服务器需要按照以下步骤进行操作: 1. 首先,需要配置LwIP和FreeRTOS。可以在STM32CubeMX中选择配置相应的组件,生成对应的代码和初始化函数。 2. 在代码中创建任务来处理TCP服务器。通过创建一个任务,可以将其分配给特定的核心,以处理TCP请求和响应。 3. 在任务中,首先需要进行LwIP和FreeRTOS的初始化。这样可以确保网络和操作系统的适当设置。需要调用lwip_init()和vTaskStartScheduler()函数。 4. 配置和创建TCP服务器的套接字。可以通过调用lwip_socket()函数创建一个TCP套接字,并使用lwip_bind()函数将其与特定的IP地址和端口绑定。 5. 通过调用lwip_listen()函数监听TCP套接字,等待客户端的连接。 6. 使用lwip_accept()函数接受客户端的连接请求,并获得一个新的套接字来处理与该客户端之间的通信。 7. 通过调用lwip_recv()和lwip_send()函数来接收和发送数据。可以使用这些函数接收来自客户端的数据,并发送响应数据给客户端。 8. 当与客户端的通信完成后,使用lwip_close()函数关闭套接字。 9. 循环进行步骤6-8,以处理其他客户端的连接和通信请求。 需要注意的是,STM32系列芯片的内存和处理能力有限,因此在编写代码时需要谨慎处理内存和资源的分配和释放,以确保程序的稳定性和性能。 总结:通过以上步骤,可以在STM32上使用FreeRTOS和LwIP实现TCP服务器,使其能够接受和处理客户端的连接和通信请求。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值