Toggle navigation
首页
问答
文章
积分商城
专家
专区
更多专区...
文档中心
返回主站
搜索
提问
会员
中心
登录
注册
netutils
wiznet
WIZnet_W5500
使用WIZ包驱动w5500传输速率很低
发布于 2023-03-02 17:53:31 浏览:1148
订阅该版
使用menuconfig配置了w5500的驱动,开启了SPI+DMA并且在IDE中使用了Release配置,但使用iperf工具或之间使用tcp程序进行速度测试,包括在任务管理器中看到的速率都很低。 iperf的输出: ``` [680] 9.0-10.0 sec 168 KBytes 1376 Kbits/sec ``` rtconfig的配置: ```c #define RT_USING_PIN #define RT_USING_SPI #define RT_USING_SAL #define SAL_INTERNET_CHECK #define SAL_USING_POSIX #define RT_USING_NETDEV #define PKG_USING_NETUTILS #define PKG_NETUTILS_IPERF #define PKG_NETUTILS_TELNET #define PKG_USING_NETUTILS_LATEST_VERSION #define PKG_NETUTILS_VER_NUM 0x99999 #define PKG_USING_WIZNET #define WIZ_USING_W5500 #define WIZ_USING_PING #define PKG_USING_WIZNET_V200 #define SOC_STM32H743II #define BSP_USING_W5500 #define BSP_USING_SPI #define BSP_USING_SPI4 #define BSP_SPI4_TX_USING_DMA ``` --- 分割线 我初步找到问题了,是STM32H7的SPI4相关时钟配置存在问题。之前认为SPI框架是成熟的,没有怀疑过时钟频率,今天用逻辑分析仪测了SCK频率只有3M左右的频率,原因是SPI驱动对时钟源取值时没有考虑时钟源Mux选择器的情况,需要手动修改时钟频率。 [stm32h7系列的spi4和spi5的分频系数计算对吗?](https://club.rt-thread.org/ask/question/264268836db4c5fd.html "stm32h7系列的spi4和spi5的分频系数计算对吗?") 对此我添加了下面的代码 ```c static uint32_t stm32_spi_get_clock(SPI_HandleTypeDef *spi_handle) { SPI_TypeDef *SPIx = spi_handle->Instance; #if defined(SOC_SERIES_STM32F0) || defined(SOC_SERIES_STM32G0) return HAL_RCC_GetPCLK1Freq(); #elif defined(SOC_SERIES_STM32H7) if (SPIx == SPI1 || SPIx == SPI2 || SPIx == SPI3) { return HAL_RCCEx_GetPeriphCLKFreq(RCC_PERIPHCLK_SPI123); } else if (SPIx == SPI4 || SPIx == SPI5) { uint32_t sel = __HAL_RCC_GET_SPI45_SOURCE(); if (sel == RCC_SPI45CLKSOURCE_D2PCLK1) { return HAL_RCC_GetPCLK2Freq(); } else if (sel == RCC_SPI45CLKSOURCE_PLL2) { PLL2_ClocksTypeDef PLL2_Clocks; HAL_RCCEx_GetPLL2ClockFreq(&PLL2_Clocks); return PLL2_Clocks.PLL2_Q_Frequency; } else if (sel == RCC_SPI45CLKSOURCE_PLL3) { PLL3_ClocksTypeDef PLL3_Clocks; HAL_RCCEx_GetPLL3ClockFreq(&PLL3_Clocks); return PLL3_Clocks.PLL3_Q_Frequency; } else if (sel == RCC_SPI45CLKSOURCE_HSI) { return HSI_VALUE; } else if (sel == RCC_SPI45CLKSOURCE_CSI) { return CSI_VALUE; } else if (sel == RCC_SPI45CLKSOURCE_HSE) { return HSE_VALUE; } } else { return HAL_RCCEx_GetPeriphCLKFreq(RCC_PERIPHCLK_SPI6); } #else return HAL_RCC_GetPCLK2Freq(); #endif return HAL_RCC_GetSysClockFreq(); } ``` 现在iperf测试相对正常了,但该测试只对PC向MCU的下行速率进行了测试。 我编写了MCU向PC上传数据的测试代码,测试结果还是不理想,只有2.1Mbps。经过测量,发现SPI通讯线每隔8ms左右才发送一次2048字节的包,两个包中间分散着少量通讯,可能是在进行忙状态查询,不知道如何优化。 ```c #include
#include
#define BUFSZ (2048) /* 发送用到的数据 */ static const char send_data1[] = "AD7980 Ready\n"; static const char send_data2[] = "AD7980 NotFound\n"; #define TEST_SIZE 10000000 // 10MB static char test_buf[BUFSZ]; static const char *send_data; static void tcpserv(void *param) { /* ---------------------AD7980 Data--------------------- */ rt_device_t drv_adc = rt_device_find("ad7980"); if (drv_adc == RT_NULL) send_data = send_data2; send_data = send_data1; for (int i = 0; i < BUFSZ; i++) test_buf[i] = i; /* ---------------------Completion--------------------- */ rt_completion_init(&cpl_rx); rt_device_set_rx_indicate(drv_adc, ad7980_rx_ind); char *recv_data; /* 用于接收的指针,后面会做一次动态分配以请求可用内存 */ socklen_t sin_size; int sock, connected, bytes_received; struct sockaddr_in server_addr, client_addr; rt_bool_t stop = RT_FALSE; /* 停止标志 */ int ret; recv_data = rt_malloc(BUFSZ + 1); /* 分配接收用的数据缓冲 */ if (recv_data == RT_NULL) { rt_kprintf("No memory\n"); return; } /* 一个socket在使用前,需要预先创建出来,指定SOCK_STREAM为TCP的socket */ if ((sock = socket(AF_INET, SOCK_STREAM, 0)) == -1) { /* 创建失败的错误处理 */ rt_kprintf("Socket error\n"); /* 释放已分配的接收缓冲 */ rt_free(recv_data); return; } /* 初始化服务端地址 */ server_addr.sin_family = AF_INET; server_addr.sin_port = htons(5000); /* 服务端工作的端口 */ server_addr.sin_addr.s_addr = INADDR_ANY; rt_memset(&(server_addr.sin_zero), 0, sizeof(server_addr.sin_zero)); /* 绑定socket到服务端地址 */ if (bind(sock, (struct sockaddr *)&server_addr, sizeof(struct sockaddr)) == -1) { /* 绑定失败 */ rt_kprintf("Unable to bind\n"); /* 释放已分配的接收缓冲 */ rt_free(recv_data); return; } /* 在socket上进行监听 */ if (listen(sock, 5) == -1) { rt_kprintf("Listen error\n"); /* release recv buffer */ rt_free(recv_data); return; } rt_kprintf("\nTCPServer Waiting for client on port 5000...\n"); while (stop != RT_TRUE) { sin_size = sizeof(struct sockaddr_in); /* 接受一个客户端连接socket的请求,这个函数调用是阻塞式的 */ connected = accept(sock, (struct sockaddr *)&client_addr, &sin_size); /* 返回的是连接成功的socket */ if (connected < 0) { rt_kprintf("accept connection failed! errno = %d\n", errno); continue; } /* 接受返回的client_addr指向了客户端的地址信息 */ rt_kprintf("I got a connection from (%s , %d)\n", inet_ntoa(client_addr.sin_addr), ntohs(client_addr.sin_port)); /* 客户端连接的处理 */ while (1) { /* 发送数据到connected socket */ ret = send(connected, send_data, strlen(send_data), 0); if (ret < 0) { /* 发送失败,关闭这个连接 */ closesocket(connected); rt_kprintf("\nsend error,close the socket.\r\n"); break; } else if (ret == 0) { /* 打印send函数返回值为0的警告信息 */ rt_kprintf("\n Send warning,send function return 0.\r\n"); } int sz = TEST_SIZE; // 10MB rt_tick_t t0 = rt_tick_get_millisecond(); while (sz > 0) { /* 发送数据到connected socket */ ret = send(connected, test_buf, sizeof(test_buf), 0); if (ret < 0) { /* 发送失败,关闭这个连接 */ closesocket(connected); rt_kprintf("\nsend error,close the socket.\r\n"); break; } else if (ret == 0) { /* 打印send函数返回值为0的警告信息 */ rt_kprintf("\n Send warning,send function return 0.\r\n"); } sz -= BUFSZ; } t0 = rt_tick_get_millisecond() - t0; rt_kprintf("\n Send finish, time: %d s, avgspd: %d KB/s\r\n", (t0 + 500) / 1000, TEST_SIZE / t0); /* 从connected socket中接收数据,接收buffer是1024大小,但并不一定能够收到1024大小的数据 */ bytes_received = recv(connected, recv_data, BUFSZ, 0); if (bytes_received < 0) { /* 接收失败,关闭这个connected socket */ closesocket(connected); break; } else if (bytes_received == 0) { /* 打印recv函数返回值为0的警告信息 */ rt_kprintf("\nReceived warning,recv function return 0.\r\n"); closesocket(connected); break; } /* 有接收到数据,把末端清零 */ recv_data[bytes_received] = '\0'; if (strcmp(recv_data, "q") == 0 || strcmp(recv_data, "Q") == 0) { /* 如果是首字母是q或Q,关闭这个连接 */ closesocket(connected); break; } else if (strcmp(recv_data, "exit") == 0) { /* 如果接收的是exit,则关闭整个服务端 */ closesocket(connected); stop = RT_TRUE; break; } else { /* 在控制终端显示收到的数据 */ rt_kprintf("RECEIVED DATA = %s \n", recv_data); } } } /* 退出服务 */ closesocket(sock); /* 释放接收缓冲 */ rt_free(recv_data); return; } ``` 输出结果 `Send finish, time: 39 s, avgspd: 250 KB/s`
查看更多
Ryan_CW
认证专家
2023-03-02
这家伙很懒,什么也没写!
RyanW5500在软件层级没有对W5500速度进行过优化 刚试了下stm32F407芯片作为服务端tcp速度测试, 使用rtthread spi驱动框架,设置最高60M spi速度(实际由于f407速度限制,可能在42m) 没有使用dma,速度如下几乎是跑满spi速度了 ![screenshot_image.png](https://oss-club.rt-thread.org/uploads/20230302/687160a4de43423b88bd8ff46acda055.png)
2
个回答
默认排序
按发布时间排序
Tamap
2023-03-20
这家伙不懒,签名都写了
问题已经解决,原因有二: 1. 该包中所有的通讯都是阻塞的且经常重复调用函数链访问W5500的寄存器。关闭多余的传输和检测,使用中断方式,实现了比较干净的通讯。 2. STM32H7的SPI驱动没有对用户当前的时钟配置进行正确识别,通过测量SCK后才发现SPI配置存在问题。其通讯速率一定程度上限制了通讯速率的上限,使其降低到原本的八分之一。 经过优化,现在可以在约30MHz的SPI速率下达到20Mbps的通讯速率,TCP约2MB/s。
撰写答案
登录
注册新账号
关注者
0
被浏览
1.1k
关于作者
Tamap
这家伙不懒,签名都写了
提问
5
回答
11
被采纳
1
关注TA
发私信
相关问题
1
W5500软件包开机无法解析link.rt-thread.org
2
W5500 如何实现断开重连?
3
W5500初始化全部都OK,电脑能ping板子,板子ping不通电脑
4
w5500 编译错误
5
求个SPI上挂W5500、Max6675两个设备的驱动例子
6
TFTP与W5500配合能传文件但无内容
7
wiznet(W5500) 设置MAC
8
SPI总线挂接2个W5500以太网芯片
9
多网卡建立链路问题,求官方解答
10
为什么W5500一直显示无法挂载
推荐文章
1
RT-Thread应用项目汇总
2
玩转RT-Thread系列教程
3
国产MCU移植系列教程汇总,欢迎查看!
4
机器人操作系统 (ROS2) 和 RT-Thread 通信
5
五分钟玩转RT-Thread新社区
6
【技术三千问】之《玩转ART-Pi》,看这篇就够了!干货汇总
7
关于STM32H7开发板上使用SDIO接口驱动SD卡挂载文件系统的问题总结
8
STM32的“GPU”——DMA2D实例详解
9
RT-Thread隐藏的宝藏之completion
10
【ART-PI】RT-Thread 开启RTC 与 Alarm组件
最新文章
1
单片机也能聊天?RT-Thread上跑通大语言模型
2
【RT-Thread】【ci】【scons】将ci.attachconfig.yml和scons结合使用
3
Rt-thread中OTA下载后,bootloader不搬程序
4
ulog 日志 LOG_HEX 输出时间改为本地日期时间
5
在RT-Thread Studio中构建前执行python命令
热门标签
RT-Thread Studio
串口
Env
LWIP
SPI
AT
Bootloader
Hardfault
CAN总线
FinSH
ART-Pi
USB
DMA
文件系统
RT-Thread
SCons
RT-Thread Nano
线程
MQTT
STM32
RTC
FAL
rt-smart
I2C_IIC
ESP8266
UART
WIZnet_W5500
ota在线升级
cubemx
PWM
flash
freemodbus
BSP
packages_软件包
潘多拉开发板_Pandora
定时器
ADC
flashDB
GD32
socket
编译报错
中断
Debug
rt_mq_消息队列_msg_queue
SFUD
msh
keil_MDK
ulog
C++_cpp
MicroPython
本月问答贡献
出出啊
1518
个答案
343
次被采纳
小小李sunny
1444
个答案
290
次被采纳
张世争
813
个答案
177
次被采纳
crystal266
547
个答案
161
次被采纳
whj467467222
1222
个答案
149
次被采纳
本月文章贡献
出出啊
1
篇文章
5
次点赞
小小李sunny
1
篇文章
1
次点赞
张世争
1
篇文章
3
次点赞
crystal266
2
篇文章
2
次点赞
whj467467222
2
篇文章
2
次点赞
回到
顶部
发布
问题
分享
好友
手机
浏览
扫码手机浏览
投诉
建议
回到
底部