赞
踩
这个是全网最详细的STM32项目教学视频。
第一篇在这里:
视频在这里
STM32智能小车V3-STM32入门教程-openmv与STM32循迹小车-stm32f103c8t6-电赛 嵌入式学习 PID控制算法 编码器电机 跟随
先通过串口上位机模拟发送、
STM32有视觉循迹模式、该模式下接收到数据根据状态显示在屏幕上,现在此状态并不控制电机。
复制一下18在上面基础改,命名成19-4_LED
可以先复制到桌面英文路径,防止出现中文路径兼容问题。
看原理图摄像头是预留什么引脚
PCB中可以看到接口位置
所以我们要初始化一下串口二,然后开启串口接收中断
串口2 开启初始化
开启串口中断
生成代码
打开代码
如果发现18章代码经常出现黑屏,那可能就是6050的初始化卡住了,我们可以注释掉一下MPU6050部分的代码。
我们先定义一个串口二接收数据变量
uint8_t g_ucUsart2ReceiveData; //保存串口二接收的数据
开启接收中断
HAL_UART_Receive_IT(&huart2,&g_ucUsart2ReceiveData,1); //串口二接收数据
声明一下变量
extern uint8_t g_ucUsart2ReceiveData; //保存串口二接收的数据
我们需要在串口中断回调函数中加入我们对接收到数据的解析
if(huart == &huart2)//判断中断源 是否来自串口二
{
//这里增加解析函数
HAL_UART_Receive_IT(&huart2,&g_ucUsart2ReceiveData,1); //启动串口二接收数据
}
在usart.c文件中定义一个函数
/******************* * @brief 摄像头串口协议解析函数 可以连接K210或openmv等 * @param data:串口接收到的每个字节 * @return * *******************/ void usartCamera_Receive_Data(uint8_t data) { static uint8_t state = 0;//定义静态static 变量 if(state==0&&data==0xA5) //判断第一个是不是帧头0xA5 { state=1;//是帧头0xA5 赋值state=1 表示接收下一个数据 //数据存储在数组中 "g_ucUsart2ReceivCounter++",这里是先用后加,比如g_ucUsart2ReceivCounter 初值为0,执行这个是先g_ucaUsart2ReceiveBuffer[0]=data,然后g_ucUsart2ReceivCounter++,即后g_ucUsart2ReceivCounter = 1的 g_ucaUsart2ReceiveBuffer[g_ucUsart2ReceivCounter++] = data; } else if(state==1&&data==0xA6) //第二个是不是帧头0xA6 { state=2;//如果第二个是帧头0xA6 赋值state=2 表示接收下一个数据 g_ucaUsart2ReceiveBuffer[g_ucUsart2ReceivCounter++] = data;//保存数据 } else if(state==2)//然后确定开头是0XA5 0XA6 就开始接收 { g_ucaUsart2ReceiveBuffer[g_ucUsart2ReceivCounter++]=data; if(g_ucUsart2ReceivCounter>9||data==0x5B) state=3; //接收大于9个或者接收到帧尾0X5B 就置位状态三 } else if(state==3) //状态三 { if(g_ucaUsart2ReceiveBuffer[g_ucUsart2ReceivCounter-1] == 0x5B) //确定 最后一个是不是0x5B帧尾 是帧尾0x5B 就认为通信正确 处理数据 { state = 0; //这里就可以处理数据了、处理完记得清空数组和重置标志位与计数值 g_ucUsart2ReceivCounter = 0;//清零计数值 //比如根据数据设置红外旋转偏移状态 //1.设置快速 慢速右边 左边 数字存储的变量意义: [0]和[1]:帧头、[2]:摄像头左边数第一个感兴趣区域、[3]:左边第二个、[4]:左边第三个、[5]:左边第四个、[6]:左边第五个、[7]:帧尾 if(g_ucaUsart2ReceiveBuffer[6]==0&&g_ucaUsart2ReceiveBuffer[5]==0&&g_ucaUsart2ReceiveBuffer[3]==0&&g_ucaUsart2ReceiveBuffer[2]==0) { g_cThisState=0;//前进 g_lHW_State=22222;//设置这个显示在OLED上方便调试 五个值 以此从左向右表示 从左向右的五个区域 } if(g_ucaUsart2ReceiveBuffer[6]==0&&g_ucaUsart2ReceiveBuffer[5]==1&&g_ucaUsart2ReceiveBuffer[3]==0&&g_ucaUsart2ReceiveBuffer[2]==0) { g_cThisState=-1;//应该右转 g_lHW_State=22212; //表示右数第二个 识别到线 } if(g_ucaUsart2ReceiveBuffer[6]==1&&g_ucaUsart2ReceiveBuffer[5]==0&&g_ucaUsart2ReceiveBuffer[3]==0&&g_ucaUsart2ReceiveBuffer[2]==0) {g_cThisState=-2;//快速右转 g_lHW_State=22221; } if(g_ucaUsart2ReceiveBuffer[6]==1&&g_ucaUsart2ReceiveBuffer[5]==1&&g_ucaUsart2ReceiveBuffer[3]==0&&g_ucaUsart2ReceiveBuffer[2]==0) {g_cThisState=-3;//快速右转 g_lHW_State=22211; } if(g_ucaUsart2ReceiveBuffer[6]==0&&g_ucaUsart2ReceiveBuffer[5]==0&&g_ucaUsart2ReceiveBuffer[3]==1&&g_ucaUsart2ReceiveBuffer[2]==0) {g_cThisState=1;//应该左转 g_lHW_State=21222; } if(g_ucaUsart2ReceiveBuffer[6]==0&&g_ucaUsart2ReceiveBuffer[5]==0&&g_ucaUsart2ReceiveBuffer[3]==0&&g_ucaUsart2ReceiveBuffer[2]==1) {g_cThisState=2;//快速左转 g_lHW_State=12222; } if(g_ucaUsart2ReceiveBuffer[6]==0&&g_ucaUsart2ReceiveBuffer[5]==0&&g_ucaUsart2ReceiveBuffer[3]==1&&g_ucaUsart2ReceiveBuffer[2]==1) {g_cThisState=3;//快速左转 g_lHW_State=11222; } //2.然后清空数组 for(int i=0;i<10;i++) g_ucaUsart2ReceiveBuffer[i]=0x00;//清空数组 } else //不是帧尾说明通信错误重新开始接收 { state=0; g_ucUsart2ReceivCounter =0; for(int i=0;i<10;i++) g_ucaUsart2ReceiveBuffer[i]=0x00;//清空数组 } } else { //其他异常清空 state=0; g_ucUsart2ReceivCounter =0; for(int i=0;i<10;i++) g_ucaUsart2ReceiveBuffer[i]=0x00;//清空数组 } }
然后声明一下变量
extern int8_t g_cThisState ;//这次状态
定义一个变量 并且在main文件中声明一下
int g_lHW_State = 0;//帮助视觉调试 用于表示红外对管或者视觉摄像头识别状态
声明一下
extern int g_lHW_State;//帮助视觉调试 用于表示红外对管或者视觉摄像头识别状态
我们需要再定义模式,这个模式是视觉循迹模式
视觉模式下 我们显示一下,我们之前赋值的变量 以测试我们接收的数据是否正确。
//这里编写触发中断后要执行的程序
if(g_ucMode == 6) g_ucMode = 1;//g_ucMode模式是0 1 2 3 4 5 6
else
{
g_ucMode+=1;
}
增加模式6,的功能,我们先只显示视觉识别结果
if(g_ucMode == 6)
{
sprintf((char*)OledString, "lHW:%d ", g_lHW_State);//视觉识别结果
OLED_ShowString(0,0,OledString,12);//这个是oled驱动里面的,是显示位置的一个函数,
motorPidSetSpeed(0,0);//停住电机防止乱跑 方便调试
}
别忘记我们的解析函数,加到串口中断处理函数中
usartCamera_Receive_Data(g_ucUsart2ReceiveData);
修改上面程序经过测试,单片机
编译上面程序,并烧录到我们的单片机、单片机连接到电脑、然后电脑模拟openmv发送正确格式的数据,手动点击SSCOM发送数据、单片机可以接收到数据并显示在OLED上(观察的是OLED的第一行数值变化)、当我们设置每1ms发送一次数据时候,单片机的OLED有时候会出现卡死的情况。所以是单片机串口接收大量数据卡死的情况,经过网上搜索发现解决问题的办法。
**这个博客是搜索到可以解决问题的链接:**https://blog.csdn.net/qq_44629109/article/details/131002223
参考博客如下部分:
所以我们要更改如下代码:
__HAL_UART_ENABLE_IT(&huart2, UART_IT_ERR);// 启用UART2的错误中断功能
在USART.C 中添加如下代码
/* UART 错误回调函数 处理串口错误 */
void HAL_UART_ErrorCallback(UART_HandleTypeDef *huart)
{
if(__HAL_UART_GET_FLAG(huart,UART_FLAG_ORE) != RESET) //使用__HAL_UART_GET_FLAG宏检查UART的overrun错误标志位是否被置位。如果返回值不等于RESET,表示overrun错误标志位被置位,即发生了overrun错误
{
__HAL_UART_CLEAR_OREFLAG(huart);//使用__HAL_UART_CLEAR_OREFLAG宏清除UART的overrun错误标志位
HAL_UART_Receive_IT(&huart2,&g_ucUsart2ReceiveData,1); //使用HAL库函数启动UART2接收中断,并设置接收缓冲区的大小为1字节
}
}
添加串口2接收变量的声明
extern uint8_t g_ucUsart2ReceiveData; //保存串口二接收的数据
让单片机处于模式6(按六下 KEY1)
上面我们测试通过上位机发送数据,然后观察屏幕。
然后我们把STM32底板接到openmv,openmv连接电脑,openmv使用的程序是19章3节的程序19-3-openmv
然后上面如果没有问题,就可以把openmv 程序通过"将打开的脚本保存到openmv Cam(作为main.py)"
接法如下:
这里就说明了如何接受的数据,后面的19.5讲解利用数据
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。