STM32f4日記4之HMI智能串口屏與單片機串口通信實驗(能通過按下串口屏圖片來控制單片機上LED燈閃滅)

STM32f4日記4之HMI智能串口屏與單片機串口通信實驗(能通過按下串口屏圖片來控制單片機上LED燈閃滅)

板子:stm32f407zgt6正點原子迷你版
串口屏:淘晶馳HMI電容觸摸屏3.5寸
//


如果你覺得對你有幫助,請點贊同,這對我很重要,謝謝。
//

一、器材介紹
HMI智能串口屏
在這裏插入圖片描述

在這裏插入圖片描述
串口屏的圖形化編輯是開發利器,參數如上
串口屏可以通過***簡單的編輯***達到非常理想的效果

二、硬件連接
串口屏一共4個口:RX、TX、5V、GND
1.單片機初始化USART1,對應正點原子板子上PA9、PA10。
2.因爲兩個IO口都是用跳線帽接好的,我們將跳線帽拔掉,將RX接PA9,將TX接PA10(接線非常重要,PA10爲單片機RX口,PA9爲單片機TX口)
3.記得設置單片機跟串口屏的波特率相同,我將他們兩個都設置爲9600(推薦串口屏使用SD卡下載,2秒就能下好,而串口下載需要2分鐘)



三、過程講解
1.想要兩個電子產品之間產生交互一定要有以下過程:發送數據,數據分析,代碼執行
2.關於發送數據串口屏可通過printh發送(此代碼發送的是hex數據也就是16進制數據)
3.根據正點原子的串口通信協議,爲保證接受數據的真實有效編寫了一個通信協議


	if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET)  
	{
   
   
		Res =USART_ReceiveData(USART1);//(USART1->DR);	
		
		if((USART_RX_STA&0x8000)==0)
		{
   
   
			if(USART_RX_STA&0x4000)
			{
   
   
				if(Res!=0x0a)USART_RX_STA=0;
				else USART_RX_STA|=0x8000;	
			}
			else 
			{
   
   	
				if(Res==0x0d)USART_RX_STA|=0x4000;
				else
				{
   
   
					USART_RX_BUF[USART_RX_STA&0X3FFF]=Res ;
					USART_RX_STA++;
					if(USART_RX_STA>(USART_REC_LEN-1))USART_RX_STA=0;  
				}		 
			}
		}   		 
  } 

定義了一個變量USART_RX_STA來進行串口通信標誌的分析,定義USART_RX_BUF(數組)來存儲收到數據
如有不懂可以觀看視頻點擊此處
這個視頻會具體講解這段代碼
這段代碼說明只有收到的數據爲0x0d 0x0a結尾纔是有效數據


四、代碼示例
usart.c裏面代碼

#include "sys.h"
#include "usart.h"	
#include"delay.h"
// 	 

#if SYSTEM_SUPPORT_OS
#include "includes.h"					 
#endif

 
#if 1
#pragma import(__use_no_semihosting)             
                
struct __FILE 
{
   
    
	int handle; 
}; 

FILE __stdout;       

void _sys_exit(int x) 
{
   
    
	x = x; 
} 

int fputc(int ch, FILE *f)
{
   
    	
	while((USART1->SR&0X40)==0);//Ñ­»··¢ËÍ,Ö±µ½·¢ËÍÍê±Ï   
	USART1->DR = (u8) ch;      
	return ch;
}
#endif
 
#if EN_USART1_RX  

 	
u8 USART_RX_BUF[USART_REC_LEN];   
u16 USART_RX_STA=0;      


void uart_init(u32 bound){
   
   
  
  GPIO_InitTypeDef GPIO_InitStructure;
	USART_InitTypeDef USART_InitStructure;
	NVIC_InitTypeDef NVIC_InitStructure;
	
	RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA,ENABLE); 
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1,ENABLE);
 
	
	GPIO_PinAFConfig(GPIOA,GPIO_PinSource9,GPIO_AF_USART1); 
	GPIO_PinAFConfig(GPIOA,GPIO_PinSource10,GPIO_AF_USART1); 
	
	
  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9 | GPIO_Pin_10; 
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;	
	GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;
	GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP;
	GPIO_Init(GPIOA,&GPIO_InitStructure);
	USART_InitStructure.USART_BaudRate = bound;
	USART_InitStructure.USART_WordLength = USART_WordLength_8b;
	USART_InitStructure.USART_StopBits = USART_StopBits_1;
	USART_InitStructure.USART_Parity = USART_Parity_No;
	USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;
	USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;	
  USART_Init(USART1, &USART_InitStructure); 
	
  USART_Cmd(USART1, ENABLE);  
	
	
	
#if EN_USART1_RX	
	USART_ITConfig(USART1, USART_IT_RXNE, ENABLE);
  NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn;//´®¿Ú1ÖжÏͨµÀ
	NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=3;
	NVIC_InitStructure.NVIC_IRQChannelSubPriority =3;		
	NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;			
	NVIC_Init(&NVIC_InitStructure);	

#endif
	
}


void USART1_IRQHandler(void)                	
{
   
   
	u8 Res;
#if SYSTEM_SUPPORT_OS 		
	OSIntEnter();    
#endif
	if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET) 
	{
   
   
		Res =USART_ReceiveData(USART1);//(USART1->DR);	
		
		if((USART_RX_STA&0x8000)==0)
		{
   
   
			if(USART_RX_STA&0x4000)//½ÓÊÕµ½ÁË0x0d
			{
   
   
				if(Res!=0x0a)USART_RX_STA=0;
				else USART_RX_STA|=0x8000;	
			}
			else 
			{
   
   	
				if(Res==0x0d)USART_RX_STA|=0x4000;
				else
				{
   
   
					USART_RX_BUF[USART_RX_STA&0X3FFF]=Res ;
					USART_RX_STA++;
					if(USART_RX_STA>(USART_REC_LEN-1))USART_RX_STA=0;
				}		 
			}
		}   		 
  } 
#if SYSTEM_SUPPORT_OS 	
	OSIntExit();  											 
#endif
} 
#endif	

 




最重要的是void USART1_IRQHandler(void) 這個函數用來寫usart1的中斷,中斷裏面就是通行協議
usart.h裏面

#define USART_REC_LEN  			200  
#define EN_USART1_RX 			1		
	  	
extern u8  USART_RX_BUF[USART_REC_LEN]; 
extern u16 USART_RX_STA;         		
void uart_init(u32 bound);
#endif

主函數裏面代碼

#include "sys.h"
#include "delay.h"
#include "usart.h"
#include "led.h"
#include "beep.h"
#include "key.h"


int main(void)
{
   
    
 
	u8 t;
	u8 len;	
	u16 times=0;  
	NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);
	delay_init(168);		
	uart_init(9600);	
	LED_Init();		  		
	while(1)
	{
   
   
		if(USART_RX_STA&0x8000)
		{
   
   					   
			len=USART_RX_STA&0x3fff;
			USART_RX_STA=0;
			
if(!(strcmp(USART_RX_BUF,"1"))) 
			{
   
   
			    LED0=0;
			}
      if(!(strcmp(USART_RX_BUF,"2"))) 
			{
   
   
			    LED0=1;
			}
		
			delay_ms(10);   
		}
	}
}


下面給出串口屏代碼編寫
在這裏插入圖片描述
在這裏插入圖片描述
這兩個圖片導入串口屏工程中
一個作爲開燈一個作爲關燈
設置一個***雙態按鈕***來實現
按下事件裏面寫





if(bt0.val==1)
{
   
   
  bt0.pic=1
  printh 31
  printh 0d
  printh 0a
}else
{
   
   
  bt0.pic=0
  printh 32
    printh 0d
  printh 0a
}

記得在program.s裏面加入

bauds=9600
bkcmd=3
dims=100

來確保波特率爲9600
下面是調試視頻我放在網盤裏面了(被百度網盤處理過,可能不怎麼清晰,見諒)

鏈接:https://pan.baidu.com/s/1_7tiJ2xq_gpxV_PdpD2CmQ 
提取碼:clz6 
複製這段內容後打開百度網盤手機App,操作更方便哦

之後就可以進行串口通信了+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++

五、實驗效果
按下串口屏圖片,燈亮,再按,燈滅

六、思考拓展
雖然這個實驗看起來很簡單,但卻是簡單化處理矩陣鍵盤的基礎,可以自行設計各種圖片按鍵,達到非常理想的效果,需要讀者自己的探索。

作者:shawn
可諮詢QQ:965798711
2021.1.22
14:55
All rights reserved



發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章