Toggle navigation
首页
问答
文章
积分商城
专家
专区
更多专区...
文档中心
返回主站
搜索
提问
会员
中心
登录
注册
ART-Pi
Ethernet_以太网
ART-PI极简Demo1 - 工业扩展板Ethernet TCPClient
发布于 2022-04-17 00:33:36 浏览:1654
订阅该版
[tocm] ## 前言 本学期在给学生们讲授嵌入式系统课程,加上有朋友在使用ART-PI过程中遇到了些许问题,本着一并分享和交流的想法,在本论坛尝试写一部分关于ART-PI的基本使用例程。最终文档将在个人公众号发布。 本系列的内容相当基础,方便小朋友们能依葫芦画瓢,在动手过程中逐渐增加兴趣,最终深入研究,学以致用,同时也方便朋友顺利开展相关工作。 小朋友们课前来论坛阅读文章,而配置的详细解释、软硬件配合讲解等就留在课堂上解决,也算是线上线下的一种新结合方式。讲课不流于形式、工作在研发一线、理论联系实际、文档规范、为学生做好榜样,先做好自己再严格要求学生,我想,也是用心践行立德树人的一种方式吧。希望能为启发小朋友投身嵌入式领域的全国产化进程贡献自己的绵薄之力。 ## 1 工程创建 1.新建工程  2.打开LwIP与SAL  3.进入Borad设置,点击上方的3处,使能ETH  4.返回`组件`部分,在LWIP上点击右键,然后在弹出部分点击`配置项`。为了简单起见,我们先失效DHCP,使用静态IP与PC直连做测试。后续有需要,将ARI-PI通过路由器连接PC,可以随时再打开DHCP。  5.保存Setting,编译,成功。  ## 2 基本测试 1.将ART-PI通电并连接网线  2.通过设备管理器确认ART-PI与PC接口。设备管理器显示,ARI-PI的ST-LINK和串口是COM9。  3.下载程序  4.下载说明 在`applications\main.c`中,将中断向量表映射到了QSPI部分。 ```c #include "stm32h7xx.h" static int vtor_config(void) { /* Vector Table Relocation in Internal QSPI_FLASH */ SCB->VTOR = QSPI_BASE; return 0; } INIT_BOARD_EXPORT(vtor_config); ``` 如下图所示,Download选项部分表示,将生成的文件下载至外部QSPI Flash中。  5.通过XShell Serial COM9连接ART-PI   6.输入ifconfig确认网卡状态  7.确认PC的IP,并进行互ping操作。 PC上打开cmd,输入ipconfig,确认PC网卡的IP地址。若它和ART-PI不在同一网段,则自行配置即可。  8.互ping操作 - ART-PI ping PC  - PC ping ART-PI  9.其他基本命令,如在ART-PI的msh中输入list_device,会列出已经驱动的设备  ## 3 其他测试 1.官方的tcpclient代码,自行通过github下载,或者将下方的代码另存为`tcpclient_sample.c`,放入工程的applications文件夹中。 ```c /* * Copyright (c) 2006-2022, RT-Thread Development Team * * SPDX-License-Identifier: Apache-2.0 * * Change Logs: * Date Author Notes * */ /* * 程序清单:tcp 客户端 * * 这是一个 tcp 客户端的例程 * 导出 tcpclient 命令到控制终端 * 命令调用格式:tcpclient URL PORT * URL:服务器地址 PORT::端口号 * 程序功能:接收并显示从服务端发送过来的信息,接收到开头是 'q' 或 'Q' 的信息退出程序 */ #include
#include
/* 使用BSD socket,需要包含socket.h头文件 */ #include
#include
#include
#define BUFSZ 1024 static const char send_data[] = "This is TCP Client from RT-Thread."; /* 发送用到的数据 */ void tcpclient(int argc, char **argv) { int ret; char *recv_data; struct hostent *host; int sock, bytes_received; struct sockaddr_in server_addr; const char *url; int port; if (argc < 3) { rt_kprintf("Usage: tcpclient URL PORT\n"); rt_kprintf("Like: tcpclient 192.168.12.44 5000\n"); return ; } url = argv[1]; port = strtoul(argv[2], 0, 10); /* 通过函数入口参数url获得host地址(如果是域名,会做域名解析) */ host = gethostbyname(url); /* 分配用于存放接收数据的缓冲 */ recv_data = rt_malloc(BUFSZ); if (recv_data == RT_NULL) { rt_kprintf("No memory\n"); return; } /* 创建一个socket,类型是SOCKET_STREAM,TCP类型 */ if ((sock = socket(AF_INET, SOCK_STREAM, 0)) == -1) { /* 创建socket失败 */ rt_kprintf("Socket error\n"); /* 释放接收缓冲 */ rt_free(recv_data); return; } /* 初始化预连接的服务端地址 */ server_addr.sin_family = AF_INET; server_addr.sin_port = htons(port); server_addr.sin_addr = *((struct in_addr *)host->h_addr); rt_memset(&(server_addr.sin_zero), 0, sizeof(server_addr.sin_zero)); /* 连接到服务端 */ if (connect(sock, (struct sockaddr *)&server_addr, sizeof(struct sockaddr)) == -1) { /* 连接失败 */ rt_kprintf("Connect fail!\n"); closesocket(sock); /*释放接收缓冲 */ rt_free(recv_data); return; } else { /* 连接成功 */ rt_kprintf("Connect successful\n"); } while (1) { /* 从sock连接中接收最大BUFSZ - 1字节数据 */ bytes_received = recv(sock, recv_data, BUFSZ - 1, 0); if (bytes_received < 0) { /* 接收失败,关闭这个连接 */ closesocket(sock); rt_kprintf("\nreceived error,close the socket.\r\n"); /* 释放接收缓冲 */ rt_free(recv_data); break; } else if (bytes_received == 0) { /* 默认 recv 为阻塞模式,此时收到0认为连接出错,关闭这个连接 */ closesocket(sock); rt_kprintf("\nreceived error,close the socket.\r\n"); /* 释放接收缓冲 */ rt_free(recv_data); break; } /* 有接收到数据,把末端清零 */ recv_data[bytes_received] = '\0'; if (strncmp(recv_data, "q", 1) == 0 || strncmp(recv_data, "Q", 1) == 0) { /* 如果是首字母是q或Q,关闭这个连接 */ closesocket(sock); rt_kprintf("\n got a 'q' or 'Q',close the socket.\r\n"); /* 释放接收缓冲 */ rt_free(recv_data); break; } else { /* 在控制终端显示收到的数据 */ rt_kprintf("\nReceived data = %s ", recv_data); } /* 发送数据到sock连接 */ ret = send(sock, send_data, strlen(send_data), 0); if (ret < 0) { /* 接收失败,关闭这个连接 */ closesocket(sock); rt_kprintf("\nsend error,close the socket.\r\n"); rt_free(recv_data); break; } else if (ret == 0) { /* 打印send函数返回值为0的警告信息 */ rt_kprintf("\n Send warning,send function return 0.\r\n"); } } return; } MSH_CMD_EXPORT(tcpclient, a tcp client sample); ``` 2.重新编译,下载。打开网络调试助手,将PC设置为TCP Server,并启动服务。  3.在Finsh中输入`tcpclient 192.168.1.107 8080`,回车,启动ART-PI中的TCP Client  4.通过网络调试助手向ART-PI发送消息。ART-PI在收到消息后,也会向PC回复一帧消息。  ## 小结 RT-Thread官方已经将ART-PI的BSP做得很完善了,尤其是配合RT-Studio工具,给广大嵌入式开发者提供了极高的便利。 至于RT-Studio的编译过程,ART-PI的启动过程,ART-PI中将程序下载到外部QSPI的原因等,可以在最基础的LED例程中进行深入研究,后续再谈。
5
条评论
默认排序
按发布时间排序
登录
注册新账号
关于作者
lchnu
Witness, Understand, Skill
文章
10
回答
229
被采纳
88
关注TA
发私信
相关文章
1
以太网先上电再连网线通信
2
ethernet 和 use memory layout冲突
3
stm32h743的LAN8720A驱动编译错误, 不知道怎么改
4
关于在stm32F107CVT6中使用以太网芯片DM9161AEP的内存不足异常
5
stm32F107+DM9161AEP的PHY芯片 运行出现错误
6
stm32f107+dm9161的phy芯片报错eth硬件初始化失败
7
stm32f107+DM9161出现硬件初始化问题
8
基于芯片建立工程,以太网卡无法使用
9
SPI总线挂接2个W5500以太网芯片
10
STM32H743 以太网驱动 问题
推荐文章
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组件
热门标签
RT-Thread Studio
串口
Env
LWIP
SPI
AT
Bootloader
Hardfault
CAN总线
FinSH
ART-Pi
DMA
USB
文件系统
RT-Thread
SCons
RT-Thread Nano
线程
MQTT
STM32
RTC
rt-smart
FAL
I2C_IIC
UART
ESP8266
cubemx
WIZnet_W5500
ota在线升级
PWM
BSP
flash
freemodbus
packages_软件包
潘多拉开发板_Pandora
定时器
ADC
GD32
flashDB
socket
编译报错
中断
Debug
rt_mq_消息队列_msg_queue
keil_MDK
ulog
SFUD
msh
C++_cpp
MicroPython
本月问答贡献
RTT_逍遥
10
个答案
3
次被采纳
xiaorui
3
个答案
2
次被采纳
winfeng
2
个答案
2
次被采纳
三世执戟
8
个答案
1
次被采纳
KunYi
8
个答案
1
次被采纳
本月文章贡献
catcatbing
3
篇文章
5
次点赞
lizimu
2
篇文章
8
次点赞
swet123
1
篇文章
4
次点赞
Days
1
篇文章
4
次点赞
YZRD
1
篇文章
2
次点赞
回到
顶部
发布
问题
投诉
建议
回到
底部