• 方案介紹
    • 19.4-STM32接收數據-狀態(tài)顯示在屏幕
  • 附件下載
  • 推薦器件
  • 相關推薦
申請入駐 產業(yè)圖譜

19.4-STM32接收數據-狀態(tài)顯示在屏幕 openMV尋跡與小車控制 STM32F103C8T6

2024/06/25
2603
加入交流群
掃碼加入
獲取工程師必備禮包
參與熱點資訊討論

聯系方式.txt

共1個文件

19.4-STM32接收數據-狀態(tài)顯示在屏幕 openMV尋跡與小車控制 Openmv+STM32F103C8T6視覺巡線小車
這個是全網最詳細的STM32項目教學視頻。
第一篇在這里:
視頻在這里


STM32智能小車V3-STM32入門教程-openmv與STM32循跡小車-stm32f103c8t6-電賽 嵌入式學習 PID控制算法 編碼器電機 跟隨

19.4-STM32接收數據-狀態(tài)顯示在屏幕

先通過串口上位機模擬發(fā)送、

STM32有視覺循跡模式、該模式下接收到數據根據狀態(tài)顯示在屏幕上,現在此狀態(tài)并不控制電機。
在這里插入圖片描述
復制一下18在上面基礎改,命名成19-4_LED

可以先復制到桌面英文路徑,防止出現中文路徑兼容問題。
在這里插入圖片描述
在這里插入圖片描述
原理圖攝像頭是預留什么引腳
在這里插入圖片描述
PCB中可以看到接口位置
在這里插入圖片描述
所以我們要初始化一下串口二,然后開啟串口接收中斷
在這里插入圖片描述
串口2 開啟初始化

在這里插入圖片描述
開啟串口中斷
在這里插入圖片描述
生成代碼
在這里插入圖片描述
打開代碼

在這里插入圖片描述
如果發(fā)現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  攝像頭串口協(xié)議解析函數 可以連接K210或openmv等
*  @param  data:串口接收到的每個字節(jié)
*  @return  
*
*******************/
void usartCamera_Receive_Data(uint8_t data)
{
	static uint8_t state = 0;//定義靜態(tài)static 變量
	
	if(state==0&&data==0xA5) //判斷第一個是不是幀頭0xA5
	{
		state=1;//是幀頭0xA5 賦值state=1 表示接收下一個數據
		//數據存儲在數組中 "g_ucUsart2ReceivCounter++",這里是先用后加,比如g_ucUsart2ReceivCounter 初值為0,執(zhí)行這個是先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 就置位狀態(tài)三
	}
	else if(state==3) //狀態(tài)三
	{
		if(g_ucaUsart2ReceiveBuffer[g_ucUsart2ReceivCounter-1] == 0x5B)  //確定 最后一個是不是0x5B幀尾 是幀尾0x5B 就認為通信正確 處理數據
		{
			state = 0;					//這里就可以處理數據了、處理完記得清空數組和重置標志位與計數值
			g_ucUsart2ReceivCounter = 0;//清零計數值
			//比如根據數據設置紅外旋轉偏移狀態(tài)

			//1.設置快速 慢速右邊 左邊 數字存儲的變量意義: [0]和[1]:幀頭、[2]:攝像頭左邊數第一個感興趣區(qū)域、[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上方便調試 五個值 以此從左向右表示 從左向右的五個區(qū)域
				 }
				 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 ;//這次狀態(tài)

在這里插入圖片描述
定義一個變量 并且在main文件中聲明一下

int g_lHW_State = 0;//幫助視覺調試 用于表示紅外對管或者視覺攝像頭識別狀態(tài)

在這里插入圖片描述
聲明一下

extern int g_lHW_State;//幫助視覺調試 用于表示紅外對管或者視覺攝像頭識別狀態(tài)

在這里插入圖片描述
我們需要再定義模式,這個模式是視覺循跡模式

視覺模式下 我們顯示一下,我們之前賦值的變量 以測試我們接收的數據是否正確。

			//這里編寫觸發(fā)中斷后要執(zhí)行的程序
			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發(fā)送正確格式的數據,手動點擊SSCOM發(fā)送數據、單片機可以接收到數據并顯示在OLED上(觀察的是OLED的第一行數值變化)、當我們設置每1ms發(fā)送一次數據時候,單片機的OLED有時候會出現卡死的情況。所以是單片機串口接收大量數據卡死的情況,經過網上搜索發(fā)現解決問題的辦法。

**這個博客是搜索到可以解決問題的鏈接:**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錯誤標志位被置位,即發(fā)生了overrun錯誤
    {
        __HAL_UART_CLEAR_OREFLAG(huart);//使用__HAL_UART_CLEAR_OREFLAG宏清除UART的overrun錯誤標志位
        HAL_UART_Receive_IT(&huart2,&g_ucUsart2ReceiveData,1);  //使用HAL庫函數啟動UART2接收中斷,并設置接收緩沖區(qū)的大小為1字節(jié)
    }
}

在這里插入圖片描述
添加串口2接收變量的聲明

extern uint8_t g_ucUsart2ReceiveData;  //保存串口二接收的數據

在這里插入圖片描述
讓單片機處于模式6(按六下 KEY1)
在這里插入圖片描述
上面我們測試通過上位機發(fā)送數據,然后觀察屏幕。

然后我們把STM32底板接到openmv,openmv連接電腦,openmv使用的程序是19章3節(jié)的程序19-3-openmv
在這里插入圖片描述
然后上面如果沒有問題,就可以把openmv 程序通過"將打開的腳本保存到openmv Cam(作為main.py)"

接法如下:
在這里插入圖片描述
這里就說明了如何接受的數據,后面的19.5講解利用數據

聯系:Q,1930299709

  • 聯系方式.txt
    下載

推薦器件

更多器件
器件型號 數量 器件廠商 器件描述 數據手冊 ECAD模型 風險等級 參考價格 更多信息
SFH619A-X009T 1 Vishay Intertechnologies SFH619A Optocoupler, Photodarlington Output, High Gain, 300 V BVCEO

ECAD模型

下載ECAD模型
$1.5 查看
CSTNE10M0G550000R0 1 Murata Manufacturing Co Ltd Ceramic Resonator,

ECAD模型

下載ECAD模型
$0.65 查看
AFBR-2529Z 1 Foxconn Receiver, Through Hole Mount, ROHS COMPLIANT, PLASTIC, 4 PIN
$22.52 查看
意法半導體

意法半導體

意法半導體(ST)集團于1987年6月成立,是由意大利的SGS微電子公司和法國Thomson半導體公司合并而成。1998年5月,SGS-THOMSON Microelectronics將公司名稱改為意法半導體有限公司。意法半導體是世界最大的半導體公司之一,公司銷售收入在半導體工業(yè)五大高速增長市場之間分布均衡(五大市場占2007年銷售收入的百分比):通信(35%),消費(17%),計算機(16%),汽車(16%),工業(yè)(16%)。 據最新的工業(yè)統(tǒng)計數據,意法半導體是全球第五大半導體廠商,在很多市場居世界領先水平。例如,意法半導體是世界第一大專用模擬芯片和電源轉換芯片制造商,世界第一大工業(yè)半導體和機頂盒芯片供應商,而且在分立器件、手機相機模塊和車用集成電路領域居世界前列.

意法半導體(ST)集團于1987年6月成立,是由意大利的SGS微電子公司和法國Thomson半導體公司合并而成。1998年5月,SGS-THOMSON Microelectronics將公司名稱改為意法半導體有限公司。意法半導體是世界最大的半導體公司之一,公司銷售收入在半導體工業(yè)五大高速增長市場之間分布均衡(五大市場占2007年銷售收入的百分比):通信(35%),消費(17%),計算機(16%),汽車(16%),工業(yè)(16%)。 據最新的工業(yè)統(tǒng)計數據,意法半導體是全球第五大半導體廠商,在很多市場居世界領先水平。例如,意法半導體是世界第一大專用模擬芯片和電源轉換芯片制造商,世界第一大工業(yè)半導體和機頂盒芯片供應商,而且在分立器件、手機相機模塊和車用集成電路領域居世界前列.收起

查看更多

相關推薦