10 vm 添加串口_STM32的串口通信
在基礎實驗成功的基礎上,對串口的調試方法進行實踐。硬件代碼順利完成之后,對日后調試需要用到的printf重定義進行調試,固定在自己的庫函數中。
b) 初始化函數定義:
void USART_Configuration(void); //定義串口初始化函數
c) 初始化函數調用:
void UART_Configuration(void); //串口初始化函數調用
初始化代碼:
void USART_Configuration(void) //串口初始化函數
{
//串口參數初始化
USART_InitTypeDef USART_InitStructure; //串口設置恢復默認參數
//初始化參數設置
USART_InitStructure.USART_BaudRate = 9600; //波特率9600
USART_InitStructure.USART_WordLength = USART_WordLength_8b; //字長8位
USART_InitStructure.USART_StopBits = USART_StopBits_1; //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;//打開Rx接收和Tx發送功能
USART_Init(USART1, &USART_InitStructure); //初始化
USART_Cmd(USART1, ENABLE); //啟動串口
}
RCC中打開相應串口
RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1 , ENABLE);
GPIO里面設定相應串口管腳模式
//串口1的管腳初始化
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9; //管腳9
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP; //復用推挽輸出
GPIO_Init(GPIOA, &GPIO_InitStructure); //TX初始化
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10; //管腳10
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING; //浮空輸入
GPIO_Init(GPIOA, &GPIO_InitStructure); //RX初始化
d) 簡單應用:
發送一位字符
USART_SendData(USART1, 數據); //發送一位數據
while(USART_GetFlagStatus(USART1, USART_FLAG_TXE) == RESET){} //等待發送完畢
接收一位字符
while(USART_GetFlagStatus(USART1, USART_FLAG_RXNE) == RESET){} //等待接收完畢
變量= (USART_ReceiveData(USART1)); //接受一個字節
發送一個字符串
先定義字符串:char rx_data[250];
然后在需要發送的地方添加如下代碼
int i; //定義循環變量
while(rx_data!='0') //循環逐字輸出,到結束字'0'
{USART_SendData(USART1, rx_data); //發送字符
while(USART_GetFlagStatus(USART1, USART_FLAG_TXE) == RESET){} //等待字符發送完畢
i++;}
e) USART注意事項:
發動和接受都需要配合標志等待。
只能對一個字節操作,對字符串等大量數據操作需要寫函數
使用串口所需設置:RCC初始化里面打開RCC_APB2PeriphClockCmd
(RCC_APB2Periph_USARTx);GPIO里面管腳設定:串口RX(50Hz,IN_FLOATING);串口TX(50Hz,AF_PP);
f) printf函數重定義(不必理解,調試通過以備后用)
(1) 需要c標準函數:
#include "stdio.h"
(2) 粘貼函數定義代碼
#define PUTCHAR_PROTOTYPE int __io_putchar(int ch) //定義為putchar應用
(3) RCC中打開相應串口
(4) GPIO里面設定相應串口管腳模式
(6) 增加為putchar函數。
int putchar(int c) //putchar函數
{
if (c == 'n'){putchar('r');} //將printf的n變成r
USART_SendData(USART1, c); //發送字符
while(USART_GetFlagStatus(USART1, USART_FLAG_TXE) == RESET){} //等待發送結束
return c; //返回值
}
(8) 通過,試驗成功。printf使用變量輸出:%c字符,%d整數,%f浮點數,%s字符串,/n或/r為換行。注意:只能用于main.c中。
3、 NVIC串口中斷的應用
a) 目的:利用前面調通的硬件基礎,和幾個函數的代碼,進行串口的中斷輸入練習。因為在實際應用中,不使用中斷進行的輸入是效率非常低的,這種用法很少見,大部分串口的輸入都離不開中斷。
b) 初始化函數定義及函數調用:不用添加和調用初始化函數,在指定調試地址的時候已經調用過,在那個NVIC_Configuration里面添加相應開中斷代碼就行了。
c) 過程:
i. 在串口初始化中USART_Cmd之前加入中斷設置:
USART_ITConfig(USART1, USART_IT_TXE, ENABLE);//TXE發送中斷,TC傳輸完成中斷,RXNE接收中斷,PE奇偶錯誤中斷,可以是多個。
ii. RCC、GPIO里面打開串口相應的基本時鐘、管腳設置
iii. NVIC里面加入串口中斷打開代碼:
NVIC_InitTypeDef NVIC_InitStructure;//中斷默認參數
NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQChannel;//通道設置為串口1中斷
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0; //中斷占先等級0
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0; //中斷響應優先級0
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; //打開中斷
NVIC_Init(&NVIC_InitStructure); //初始化
iv. 在stm32f10x_it.c文件中找到void USART1_IRQHandler函數,在其中添入執行代碼。一般最少三個步驟:先使用if語句判斷是發生那個中斷,然后清除中斷標志位,最后給字符串賦值,或做其他事情。
void USART1_IRQHandler(void) //串口1中斷
{
char RX_dat; //定義字符變量
if (USART_GetITStatus(USART1, USART_IT_RXNE) != RESET) //判斷發生接收中斷
{USART_ClearITPendingBit(USART1, USART_IT_RXNE); //清除中斷標志
GPIO_WriteBit(GPIOB, GPIO_Pin_10, (BitAction)0x01); //開始傳輸
RX_dat=USART_ReceiveData(USART1) & 0x7F; //接收數據,整理除去前兩位
USART_SendData(USART1, RX_dat); //發送數據
while(USART_GetFlagStatus(USART1, USART_FLAG_TXE) == RESET){}//等待發送結束
}
}
d) 中斷注意事項:
可以隨時在程序中使用USART_ITConfig(USART1, USART_IT_TXE, DISABLE);來關閉中斷響應。
NVIC_InitTypeDef NVIC_InitStructure定義一定要加在NVIC初始化模塊的第一句。
全局變量與函數的定義:在任意.c文件中定義的變量或函數,在其它.c文件中使用extern+定義代碼再次定義就可以直接調用了。
STM32筆記之九:打斷它來為我辦事,EXIT (外部I/O中斷)應用
a) 目的:跟串口輸入類似,不使用中斷進行的IO輸入效率也很低,而且可以通過EXTI插入按鈕事件,本節聯系EXTI中斷。
b) 初始化函數定義:
void EXTI_Configuration(void); //定義IO中斷初始化函數
c) 初始化函數調用:
EXTI_Configuration();//IO中斷初始化函數調用簡單應用:
d) 初始化函數:
void EXTI_Configuration(void)
{ EXTI_InitTypeDef EXTI_InitStructure; //EXTI初始化結構定義
EXTI_ClearITPendingBit(EXTI_LINE_KEY_BUTTON);//清除中斷標志
GPIO_EXTILineConfig(GPIO_PortSourceGPIOA, GPIO_PinSource3);//管腳選擇
GPIO_EXTILineConfig(GPIO_PortSourceGPIOA, GPIO_PinSource4);
GPIO_EXTILineConfig(GPIO_PortSourceGPIOA, GPIO_PinSource5);
GPIO_EXTILineConfig(GPIO_PortSourceGPIOA, GPIO_PinSource6);
EXTI_InitStructure.EXTI_Mode = EXTI_Mode_Interrupt;//事件選擇
EXTI_InitStructure.EXTI_Trigger = EXTI_Trigger_Falling;//觸發模式
EXTI_InitStructure.EXTI_Line = EXTI_Line3 | EXTI_Line4; //線路選擇
EXTI_InitStructure.EXTI_LineCmd = ENABLE;//啟動中斷
EXTI_Init(&EXTI_InitStructure);//初始化
}
e) RCC初始化函數中開啟I/O時鐘
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA , ENABLE);
GPIO初始化函數中定義輸入I/O管腳。
//IO輸入,GPIOA的4腳輸入
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_4;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU; //上拉輸入
GPIO_Init(GPIOA, &GPIO_InitStructure); //初始化
f) 在NVIC的初始化函數里面增加以下代碼打開相關中斷:
NVIC_InitStructure.NVIC_IRQChannel = EXTI9_5_IRQChannel; //通道
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;//占先級
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0; //響應級
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; //啟動
NVIC_Init(&NVIC_InitStructure); //初始化
g) 在stm32f10x_it.c文件中找到void USART1_IRQHandler函數,在其中添入執行代碼。一般最少三個步驟:先使用if語句判斷是發生那個中斷,然后清除中斷標志位,最后給字符串賦值,或做其他事情。
if(EXTI_GetITStatus(EXTI_Line3) != RESET) //判斷中斷發生來源
{ EXTI_ClearITPendingBit(EXTI_Line3); //清除中斷標志
USART_SendData(USART1, 0x41); //發送字符“a”
GPIO_WriteBit(GPIOB, GPIO_Pin_2, (BitAction)(1-GPIO_ReadOutputDataBit(GPIOB, GPIO_Pin_2)));//LED發生明暗交替
}
h) 中斷注意事項:
中斷發生后必須清除中斷位,否則會出現死循環不斷發生這個中斷。然后需要對中斷類型進行判斷再執行代碼。
使用EXTI的I/O中斷,在完成RCC與GPIO硬件設置之后需要做三件事:初始化EXTI、NVIC開中斷、編寫中斷執行代碼。
Linux-C語言學習交流群【721709245】在學的進群一起交流,資料自己群文件下載
相關資料:
stm32 USART串口應用
PWM脈寬調制技術
基于STM32講解串口操作
通過Z-stack協議棧實現串口透傳
stm32直流電機驅動
《新程序員》:云原生和全面數字化實踐50位技術專家共同創作,文字、視頻、音頻交互閱讀總結
以上是生活随笔為你收集整理的10 vm 添加串口_STM32的串口通信的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 2000坐标转换成经纬度_ArcGIS中
- 下一篇: react 动态添加组件属性_这么高质量