xusiwei1236 发表于 2022-6-12 17:15

【先楫HPM6750测评】RT-Thread SPI驱动和WiFi联网

本帖最后由 xusiwei1236 于 2022-6-12 17:19 编辑

<p>本篇将会介绍&mdash;&mdash;如何使用RT-Thread Studio让HPM6750EVKMINI开发板通过RW007 WiFi模块实现联网,并介绍其背后的原理,以及使用iperf对网速进行网速测试。本篇的最后,将演示WiFi网络联通之后,将会编写一个简单联网获取百度首页的示例,并进行测试。</p>

<h2>创建RT-Thread项目</h2>

<p>开始本篇实验前,需要搭建RT-Thread开发环境,具体可以参考我之前发的文章。(可以在EEWorld我的日志页面找到:<a href="https://home.eeworld.com.cn/home.php?mod=space&amp;uid=1305707&amp;do=blog&amp;view=me&amp;from=space">xusiwei1236的日志 - 电子工程世界-论坛 (eeworld.com.cn)</a>)</p>

<p>使用RT-Thread Studio创建名为hpm_net_test的项目:</p>

<p></p>

<h2>为项目添加RW007支持</h2>

<h3>打开RT-Thread Settings</h3>

<p>项目创建成功后,打开项目的RT-Thread Settings界面:</p>

<p>可以看到,默认情况下常见的项目SPI驱动已经打开了。</p>

<p>&nbsp;</p>

<p>BSP中的SPI1驱动也已经打开了:</p>

<p></p>

<h3>添加RW007软件包</h3>

<p>在RT-Thread Settings界面,点击通过&ldquo;添加软件包&rdquo;按钮,会弹出RT-Thread Package Center界面:</p>

<p></p>

<p>在中间的搜索框种输入RW007,回车,可以找到RW007驱动程序软件包:</p>

<p></p>

<p>点击界面&ldquo;添加&rdquo;按钮,即可将RW007软件包添加到当前项目的包配置中了,此时软件包并没有真正下载下来。点完添加按钮后,界面回到了RT-Thread Settings,此时按Ctrl+S保存,则会开始下载。下载过程中,控制台子窗口中可以看到一些日志输出:</p>

<p></p>

<p>稍等片刻,可以看到控制台中间有&ldquo;RW007 v2.0.1 is downloaded successfully.&rdquo;输出。此时rw007软件包已经成功下载到当前项目中了,具体代码位于packages子目录下:</p>

<p></p>

<h3>配置RW007驱动</h3>

<p>在RT-Thread Settings界面,中将鼠标移动到RW007组件上,会弹出悬浮菜单:</p>

<p></p>

<p>点击悬浮菜单中的&ldquo;配置项&rdquo;,即可进入RW007软件包的配置界面:</p>

<p></p>

<p>可以看到,默认有一个RW007 for stm32的配置,就是说RW007默认包含了STM32的驱动。</p>

<p>这里我们需要修改的就是这个example driver port配置项,点击下拉菜单改为不使用示例驱动:</p>

<p></p>

<p>选中后,记得Ctrl+S保存配置。</p>

<p>&nbsp;</p>

<h3>编译、烧录、运行项目</h3>

<p>在RT-Thread Studio中Ctrl+B或按&ldquo;锤子&rdquo;按钮,即可开始编译项目。编译完成后,可以看到控制台输出了RAM和Flash占用:</p>

<p></p>

<p>此时,将开发板连接到PC,并使用串口助手或者其他终端工具,连接到新增的串口上。</p>

<p>再到RT-Thread Studio中,使用&ldquo;下载&rdquo;按钮或Ctrl+Alt+D即可进行烧录(或者直接进行调试也可以)。</p>

<p>烧录完成后,可以看到串口终端上有输出:</p>

<p></p>

<p>可以看到,输出了RT-Thread版本信息和RW007模组的序列号以及固件版本信息。这里能够看到RW007模组的固件版本信息,其实HPM6750芯片和RW007模组之间可以已经正常通信了。</p>

<p>&nbsp;</p>

<h2>WiFi测试</h2>

<p>接下来,我们进行一些简单的WiFi测试。</p>

<p>添加RW007组件后,默认会打开RT-Thread的WiFi驱动框架,而RT-Thread的WiFi驱动框架中同时带有一个测试命令&mdash;&mdash;wifi(对就是这么直接)。</p>

<p>我们可以在RT-Thread的finsh交互环境中使用help查看当前已有哪些命令:</p>

<p></p>

<p>可以看到有一个wifi命令。</p>

<p>&nbsp;</p>

<p>接下来我们查看wifi命令的使用方式:</p>

<p></p>

<h3>扫描测试</h3>

<p>尝试扫描周围的WiFi热点:</p>

<p>可以看到,成功扫描到了周围的WiFi热点。</p>

<p>&nbsp;</p>

<h3>连接测试</h3>

<p>尝试连接其中的一个热点:</p>

<p>然而,不幸的是,发生异常了。</p>

<p>不过,从这里的几个warning打印信息可以看到,应该是因为tcpip线程栈溢出导致的。</p>

<p>&nbsp;</p>

<h3>调大tcpip线程栈大小</h3>

<p>接下来,我们通过RT-Thread Settings修改tcpip线程栈的大小。</p>

<p>同样,首先打开RTT Settings界面,鼠标指针放到LwIP组件图标上:</p>

<p></p>

<p>打开配置项,找到<strong>RT_LWIP_TCPTHREAD_STACKSIZE</strong>配置项,并将其修改为4096:</p>

<p></p>

<p>界面下方可以看到这个LwIP线程栈大小的配置项名称为**RT_LWIP_TCPTHREAD_STACKSIZE。**至于这里为什么要改这个配置项,没有在RT-Thread用过LwIP的同学可能会疑惑。其实,这里可以根据线程名&ldquo;tcpip&rdquo;,一路搜索代码,首先可以找到创建名为tcpip线程的代码位置,然后可以找到线程栈大小参数的来源。这里是搜索结果:</p>

<p></p>

<p>PS:因为默认使用的是lwip 2.0.3版本,所以这里只搜索了lwip-2.0.3的代码。<br />
&nbsp;</p>

<h3>重新测试</h3>

<p>配置修改完成后,Ctrl+S保存,重新编译项目、烧录、运行,这次能够成功连接WiFi热点了:</p>

<p></p>

<p>可以看到,已经成功通过DHCP从热点获取到IP地址了。</p>

<p>&nbsp;</p>

<h2>网络测试</h2>

<h3>RT-Thread网络组件</h3>

<p>前面提到,添加了RW007软件包后,会开启RT-Thread的WiFi驱动框架;同时,也会开启系统中网络协议相关的组件,主要包括套接字抽象层(SAL)、网络接口层、轻量级TCP/IP堆栈(LwIP),如下图所示。</p>

<p>其中,LwIP的默认版本用的是v2.0.3,也可以切换为其他版本(RT-Thread系统中同时提供了LwIP的好几个版本可供选择)。</p>

<p>&nbsp;</p>

<h3>RT-Thread网络组件相关的命令</h3>

<p>RT-Thread系统网络相关组件打开后,将会向finsh中注册几个命令用于测试,具体包括:ifconfig、ping、netstat、dns等,可以在help的输出中找到:</p>

<p></p>

<h3>ping测试</h3>

<p>有IP地址了,我们可以用ping命令测试一下能不能访问baidu.com:</p>

<p>可以看到,能够成功ping通baidu.com了。</p>

<p>使用baidu.com的域名能够访问,说明DNS整个流程都是OK的,同时网路协议也是没问题的。</p>

<p>&nbsp;</p>

<h3>socket测试</h3>

<p>可以ping通baidu之后,我们就可以进行业务开发了。这里以一个简单的使用socket获取baidu首页为例(其实,更简单的方法是直接使用web_client组件):</p>

<pre>
<code class="language-cpp">/*
* Copyright (c) 2006-2021, RT-Thread Development Team
*
* SPDX-License-Identifier: Apache-2.0
*
* Change Logs:
* Date         Author       Notes
* 2022-05-08   xusiwei1236       the first version
*/
#include &lt;rtthread.h&gt;
#include &lt;stdlib.h&gt;
#include &lt;sys/socket.h&gt;
#include &lt;netdb.h&gt;

#define DEFAULT_HOST "example.com"
#define DEFAULT_PORT 80

#define CONTENT_LENGTH "Content-Length:"
#define HEADER_END_MARK "\\r\\n\\r\\n"

uint32_t get_host_addr(const char *host)
{
    uint32_t dest = 0;
    struct hostent *he;

    he = gethostbyname(host);
    if (he &amp;&amp; he-&gt;h_addr_list &amp;&amp; he-&gt;h_addr_list) {
      dest = ((struct in_addr *)(he-&gt;h_addr_list))-&gt;s_addr;
    }
    return dest;
}

#define close(fd) closesocket(fd)

int fetch(int argc, char* argv[])
{
    char* host = DEFAULT_HOST;
    int port = DEFAULT_PORT;
    int sockfd = -1;
    int retval = 0;
    int recved = 0;
    int content_start = 0;
    int content_length = 0;
    struct sockaddr_in server_addr = {0};
    static char request;
    static char response;

    if (argc &gt; 1) host = argv;
    if (argc &gt; 2) port = atoi(argv);

    sockfd = socket(AF_INET, SOCK_STREAM, 0);
    if (sockfd &lt; 0) {
      rt_kprintf("create socket failed!\\n");
      return -1;
    }
    rt_kprintf("create socket success!\\n");

    rt_memset(&amp;server_addr, 0, sizeof(server_addr));
    server_addr.sin_family = AF_INET;
    server_addr.sin_port = htons(port);
    server_addr.sin_addr.s_addr = get_host_addr(host);

    // IP转为 “点分十进制” 格式
    inet_ntop(AF_INET, &amp;server_addr.sin_addr, response, sizeof(response));
    rt_kprintf("server IP: %s\\n", response);

    rt_kprintf("connect to server...\\n");
    retval = connect(sockfd, (const struct sockaddr *)&amp;server_addr,
            sizeof(server_addr));
    if (retval &lt; 0) {
      rt_kprintf("connect failed!\\n");
      close(sockfd);
      return -1;
    }

    rt_snprintf(request, sizeof(request),
            "GET / HTTP/1.1\\r\\n"
            "Host: %s\\r\\n"
            "User-Agent: curl/7.81.0\\r\\n"
            "Accept: */*\\r\\n"
            "\\r\\n", host);

    rt_kprintf("send request...\\n");
    retval = send(sockfd, request, rt_strlen(request), 0);
    if (retval &lt; 0) {
      rt_kprintf("send failed!\\n");
      close(sockfd);
      return -1;
    }
    rt_kprintf("%d bytes sent\\n", retval);

    rt_kprintf("recv response...\\n");
    recved = 0;
    while ((retval = recv(sockfd, &amp;response, sizeof(response) - recved, 0)) &gt; 0) {
      if (content_length == 0) {
            char* content_length_pos = rt_strstr(response, CONTENT_LENGTH);
            if (content_length_pos) {
                content_length = atoi(content_length_pos + rt_strlen(CONTENT_LENGTH));
                rt_kprintf("found %s %d!\\n", CONTENT_LENGTH, content_length);
            }
      }
      if (content_start == 0) {
            char* header_end = rt_strstr(response, HEADER_END_MARK);
            if (header_end) {
                content_start = header_end + rt_strlen(HEADER_END_MARK) - response;
                rt_kprintf("content_start: %d\\n", content_start);
            }
      }
      recved += retval;
      rt_kprintf("recved: %d %d %d\\n", recved, content_start, content_length);
      if (content_length &amp;&amp; content_start &amp;&amp; recved - content_start &gt;= content_length) {
            rt_kprintf("fully recved!\\n");
            break;
      }
    }
    response = '\\0';

    rt_kprintf("==== Response Header ====:\\n");
    for (int i = 0; i &lt; content_start; i++) {
      rt_kprintf("%c", response);
    }

    rt_kprintf("==== Response Content ====:\\n");
    for (int i = content_start; i &lt; recved; i++) {
      rt_kprintf("%c", response);
    }

    if (retval &lt; 0) {
      rt_kprintf("recv failed!\\n");
      close(sockfd);
      return -1;
    }

    shutdown(sockfd, SHUT_RDWR);
    close(sockfd);
    return 0;
}
MSH_CMD_EXPORT(fetch, "fetch home page of a site");</code></pre>

<p>这是一段使用裸socket实现的简单HTTP客户端,依次进行了请求发送、回复接收和回复解析的过程,测试结果:</p>

<p><i><i></i></i></p>

<h2>网络带宽测试</h2>

<h3>添加netutils软件包</h3>

<p>RT-Thread的netutils组件中提供了iperf命令,可以用于测试网络带宽;</p>

<p>和前面类似的方法,为项目添加netutils组件:</p>

<p><i><i></i></i></p>

<p>打开&ldquo;配置项&rdquo;后,打开iperf的配置项:</p>

<p><i><i></i></i>修改配置后,Ctrl+S保存。</p>

<p>重新编译、烧录、运行项目,help的输出可以看到多了iperf命令。</p>

<p>&nbsp;</p>

<h3>iperf命令参数</h3>

<p>在RT-Thread的finsh中运行iperf,默认输出帮助信息:</p>

<p><i><i></i></i>可以看到iperf的命令参数使用方法。</p>

<p><strong>需要注意的是:</strong></p>

<ol>
        <li>RT-Thread的iperf命令实现中,对参数的顺序由要求,如果使用过程中发现参数报错,需要查看源码定位原因;</li>
        <li>RT-Thread的iperf不支持持续时间选项,一般是先启动,后通过stop选项停止的方式控制测试时长;</li>
</ol>

<p>&nbsp;</p>

<h3>PC端的iperf</h3>

<p>PC端的iperf可以到iperf项目官网下载:<a href="https://iperf.fr/iperf-download.php">https://iperf.fr/iperf-download.php</a></p>

<p>我使用的mobaxterm,里面自带了iperf命令,所以就不单独下载了:</p>

<p><i><i></i></i></p>

<h3>进行iperf测试</h3>

<p>进行iperf测试之前,需要注意:</p>

<ol>
        <li>最好用PC创建热点,用无线路由器也行,但是需要确保信号强度足够;</li>
        <li>确保开发板和PC直接的距离不要太远,否则WiFi信号较弱,测试的结果可能会偏小;</li>
        <li>最好在WiFi热点较少的环境下进行测试,否则测出的结果数据也会偏小;</li>
</ol>

<p>下面进行测试,测试步骤如下:</p>

<ol>
        <li>在PC上,创建热点,例如名为rtt,密码为12345678</li>
        <li>在PC上,启动iperf服务端:iperf -s -p 5678</li>
        <li>在PC上,使用ipconfig/ifconfig命令查看热点的IP地址,例如我在Win10上创建的热点,IP地址是:192.168.137.1</li>
        <li>在开发板上,连接PC启动热点:wifi join rtt 12345678</li>
        <li>在开发板上,查看IP地址是否已成功分配:ifconfig,另外,可以通过ping命令测试开发板和PC直接IP是否可达</li>
        <li>在开发板上,启动iperf客户端:iperf -c 192.168.137.1 -p 5678
        <ul>
                <li>启动后,可以通过ps命令查看正在运行的线程</li>
        </ul>
        </li>
        <li>一段时间后,在开发板上,停止iperf客户端:iperf --stop</li>
        <li>开发板上iperf停止后,PC端应该可以看到iperf的输出;</li>
</ol>

<p>开发板上整个过程的输出如下:</p>

<p><i><i></i></i></p>

<p>PC端输出:<br />
<i></i>可以看到带宽为7.45Mbps</p>

<p>&nbsp;</p>

<h3>iperf测试小结</h3>

<p>实际上,影响WiFi带宽测试结果数据的因素很多。我们这里,其中,起决定性的的主要由以下几个方面:</p>

<ol>
        <li>RW007模组本身支持的最高WiFi传输速率;</li>
        <li>RW007模组的SPI接口支持的最高工作频率;</li>
        <li>HPM6750 SPI接口最高支持的工作频率;</li>
        <li>热点(PC或路由器)的WiFi最高传输速率;</li>
        <li>各种环境因素,例如开发板和PC直接的距离、环境是否有其他热点干扰等等;</li>
</ol>

<p>&nbsp;</p>

<h2>原理简介</h2>

<p>以上操作,我们没有任何底层驱动相关代码,就实现了通过HPM6750EVKMINI开发板的RW007 WiFi模组实现联网功能。这是因为我们基于RT-Thread的项目中,从底到上已经有了:</p>

<ul>
        <li>HPM6750EVKIMNI BSP中包含了SPI驱动(libraries/drivers/drv_spi.c文件);
        <ul>
                <li>默认打开了spi1的编译配置;</li>
        </ul>
        </li>
        <li>HPM6750EVKIMNI BSP中包含了网卡初始化代码(board/rw007_port.c文件);
        <ul>
                <li>向系统注册了启动时自动执行的wifi_spi_device_init函数;</li>
                <li>wifi_spi_device_init函数内部会调用rw007软件包中的rt_hw_wifi_init函数;</li>
        </ul>
        </li>
        <li>RW007软件包,包含RW007模组的驱动代码;
        <ul>
                <li>底层使用SPI驱动实现主控和RW007模组之间的通讯;</li>
                <li>上层向RT-Thread系统注册WLAN设备(rt_hw_wifi_init函数内部会调用rt_wlan_dev_register函数);</li>
        </ul>
        </li>
        <li>RT-Thread的WiFi(也叫WLAN)驱动框架;
        <ul>
                <li>对下连接具体的 WIFI 驱动,控制 WIFI 的连接断开,扫描等操作。</li>
                <li>对上承载不同的应用,为应用提供 WIFI 控制,事件,数据导流等操作,为上层提供统一的 WIFI 控制接口。</li>
        </ul>
        </li>
        <li>RT-Thread的Socket抽象层(SAL),统一集中不同的socket实现;</li>
        <li>RT-Thread的TCP/IP协议栈(lwip+一些专有修改),具体的TCP/IP协议实现;</li>
</ul>

<p>本篇就到这里了,感谢你的阅读,下次再见。</p>

Jacktang 发表于 2022-6-13 07:15

<p>楼主经验,RT-Thread的iperf命令操作时,有顺序有要求,这个是有点麻烦,报错需要不时地检查源码查找原因</p>

xusiwei1236 发表于 2022-6-13 10:04

<p>socket测试代码文件:https://gitee.com/hpm6750/hpm_rtt_test/blob/master/applications/fetch.c</p>

xusiwei1236 发表于 2022-6-13 10:05

socket测试代码文件:https://gitee.com/hpm6750/hpm_rtt_test/blob/master/applications/fetch.c

littleshrimp 发表于 2022-6-13 14:30

<p>很详细的帖子,先收藏着,我还没用过rtt,哪天需要联网时再来仔细研究。</p>
页: [1]
查看完整版本: 【先楫HPM6750测评】RT-Thread SPI驱动和WiFi联网