您的位置:網站首頁 > 電器維修資料網 > 正文 >
STM32:玩玩修改串口
來源: 日期:2013-12-5 10:33:17 人氣:標簽:
這里的stm32_eval.c,stm32_eval.h文件,以及圖中所示三個文件夾中任意一個文件夾中的部分文件復制到第一個圖所示的文件夾中去,這里我們選擇stm3210e_eval這個文件夾。
需要復制的文件是stm3210e_eval.h
如同前面一樣建立工程,并且注意修改stm32_eval.h的內容
將//#define use_stm3210e_eval 前的#去掉。
這樣,就可以編譯并通過文件,用軟件仿真,在usart #1窗口顯示出
usart printf example: retarget the c library printf function to the usart
這樣一行字了。
顯然,對這樣的玩法我是不會滿意的,下面試著去掉與stm32e_eval等相關文件,把這里面需要用到的函數直接復制到main中去,同時,也了解一些串口設置的知識。
學到這里,多少有點明白了,stm提供的庫為了達到通用性的要求,用了很多的符號來替代常量,然后七轉八拐,有時不知要轉多少個彎才能找到 終對寄存器操作的代碼。這時,keil提供的符號瀏覽就很有用處了。方法是在將光標移到需要查看的符號上面,按下f12即可,通常可以直接跳轉到所需查看到的符號的出處。如下圖
將光標移到usart_baudrate處,按下f12,即跳轉到stm32f10x_uart.h文件中相應的定義處:
如果stm32f10x_uart.h文件沒有打開,那么這個動作會自動打開這個文件。
下面我們將eval板相關的函數復制到main函數中,以便丟掉與eval板相關的文件。
(1)打開stm32_eval.h文件,將
typedef enum
{
com1 = 0,
com2 = 1
} com_typedef;
復制到main.c中,這是用來選擇哪一個串口的,因為我的板子上也有2個串口,所以就把它復制過來,也省得對函數作較大的修改了。
(2)打開stm32_eval.c文件,有一個
void stm_eval_cominit(com_typedef com, usart_inittypedef* usart_initstruct)
的函數,是用來初始化端口的,我們把它復制到main.c中,并且把它改名為
void stm_cominit(com_typedef com, usart_inittypedef* usart_initstruct)
去掉中間的eval。
當然,在main函數中調用這個函數的地方也要做相應的修改。
這個函數中用到了如上圖中藍色框中的一些符號,又是一系列的轉換,用剛才所說的跟蹤方法,找到這些符號的原始出處,作出修改, 后得到的stm_cominit函數如下:
void stm_cominit(com_typedef com, usart_inittypedef* usart_initstruct)
{
gpio_inittypedef gpio_initstructure;
/* 打開uart所用到的gpio引腳的時鐘*/
rcc_apb2periphclockcmd(rcc_apb2periph_gpioa | rcc_apb2periph_afio, enable);
/* 打開uart的時鐘*/
if (com == com1)
{
rcc_apb2periphclockcmd(rcc_apb2periph_usart1, enable);
}
else //com=com2
{
rcc_apb1periphclockcmd(rcc_apb1periph_usart2, enable);
}
/* 配置tx引腳為推挽式輸出 */
if(com==com1)
gpio_initstructure.gpio_pin = gpio_pin_9 ;
else
gpio_initstructure.gpio_pin = gpio_pin_2;
gpio_initstructure.gpio_mode = gpio_mode_af_pp;
gpio_initstructure.gpio_speed = gpio_speed_50mhz;
gpio_init(gpioa, &gpio_initstructure);
/* 配置rx引腳為浮動輸入(高阻?) */
if(com==com1)
gpio_initstructure.gpio_pin = gpio_pin_10 ;
else
gpio_initstructure.gpio_pin = gpio_pin_3;
gpio_initstructure.gpio_mode = gpio_mode_in_floating;
gpio_init(gpioa, &gpio_initstructure);
/* 串行口配置*/
if(com==com1)
usart_init(usart1, usart_initstruct);
else
usart_init(usart2, usart_initstruct);
/* 串口允許*/
if(com==com1)
usart_cmd(usart1, enable);
else
usart_cmd(usart2, enable);
}
至此,修改基本結束,在工程中移去stm32_eval相關的各個文件,在app文件夾中將這些文件刪除,關閉工程,再重新打開工程,編譯通過,運行通過。
下面對上述初始化工作做一些解讀,當然,少不了要數據手冊的幫忙了。
(1)uart1的時鐘來源和其他串口的時鐘來源不同,uart1的時鐘來源是:apb2,其他串口的時鐘來源:apb1。
(2)用于uart通信的引腳不會自動配置,需要手工配置。其中用于輸出信號的引腳tx必須配置成為推挽式輸出,而rx引腳則配置成浮動型輸入。
(3)串口波特率、停止位等參數由庫提供的stm32f10x_usart.c中的
void usart_init(usart_typedef* usartx, usart_inittypedef* usart_initstruct)
函數來設定。
觀察這個函數的執行,可以看到函數通過對cr2寄存器的操作來設定停止位,如下圖藍色框中所示。
通過對cr1寄存器的設定來確定數據位/奇偶校驗位等,這些都只需要找到相應的符號,就能順利地進行設置,找到符號的方法,當然還是上面的按f12瀏覽的方法。
還有一個重要的工作是波特率的計算,且看這里是如何來做的。
下面這一段是波特率設置的代碼
首先根據usartxbase的值來確定需要配置的是usart1還是usart2
usartxbase = (uint32_t)usartx;
而usartx是傳入這個函數的一個參數。
然后據此來得到用于usart的時鐘頻率,這個頻率值被變量apbclock記錄。
從上面變量的跟蹤可以看到apbclock的值是0x44aa200即72000000,也就是72mhz。
接下來的一系列計算式就是根據波特率的值來計算應該傳入brr寄存器的值了,偷點懶,這里就不對算式進行一一分析了(我認為暫時沒有這個必要)。
至此,usart的設置工作完成,即完成了其數據位、停止位、奇偶校驗位、波特率的設置工作。異步通信的配置工作完成。當然,細細分析,可以發現,初始還按默認方式處理了硬件握手等的處理工作。
除了使用庫函數提供的printf等函數外,我們在單片機開發中還經常使用直接對數據寄存器賦值的方法來使用串口。stm32串口的數據寄存器名為dr,因此,我試著在main函數中寫入這樣一行:
while1()
{ usart1-》dr=0x55;
}
一試成功,軟件仿真時,在串行窗口出現了大串的字符55.
好了,串口暫時告一段落。
【看看這篇文章在百度的收錄情況】
相關文章
- 上一篇: STM32 :用PWM的方法實現熒火蟲燈
- 下一篇: STM32:用定時器實現熒火蟲燈