本文內容基于作者對固件庫的理解,按照便于理解的順序進行整理介紹,部分參考了固件庫的說明,但是也基本上重新表述并按照作者理解的順序進行重新編寫。
1.1 基于標準外設庫的軟件開發(fā)
1.1.1 STM32標準外設庫概述
STM32標準外設庫之前的版本也稱固件函數庫或簡稱固件庫,是一個固件函數包,它由程序、數據結構和宏組成,包括了微控制器所有外設的性能特征。
該函數庫還包括每一個外設的驅動描述和應用實例,為開發(fā)者訪問底層硬件提供了一個中間API,通過使用固件函數庫,無需深入掌握底層硬件細節(jié),開發(fā)者就可以輕松應用每一個外設。因此,使用固態(tài)函數庫可以大大減少用戶的程序編寫時間,進而降低開發(fā)成本。
每個外設驅動都由一組函數組成,這組函數覆蓋了該外設所有功能。每個器件的開發(fā)都由一個通用API (application programming interface 應用編程界面)驅動,API對該驅動程序的結構,函數和參數名稱都進行了標準化。
ST公司2007年10月發(fā)布了V1.0版本的固件庫,MDK ARM3.22之前的版本均支持該庫。2008年6月發(fā)布了V2.0版的固件庫,從2008年9月推出的MDK ARM3.23版本至今均使用V2.0版本的固件庫。V3.0以后的版本相對之前的版本改動較大,本書使用目前較新的V3.4版本。
1.1.2 使用標準外設庫開發(fā)的優(yōu)勢
簡單的說,使用標準外設庫進行開發(fā)最大的優(yōu)勢就在于可以使開發(fā)者不用深入了解底層硬件細節(jié)就可以靈活規(guī)范的使用每一個外設。標準外設庫覆蓋了從GPIO到定時器,再到CAN、I2C、SPI、UART和ADC等等的所有標準外設。對應的C源代碼只是用了最基本的C編程的知識,所有代碼經過嚴格測試,易于理解和使用,并且配有完整的文檔,非常方便進行二次開發(fā)和應用。
1.1.3 STM32F10XXX標準外設庫結構與文件描述
標準外設庫的文件結構
對標準外設庫的熟悉程度直接影響到程序的編寫,下面讓我們來認識一下STM32F10XXX的標準外設庫。STM32F10XXX的標準外設庫經歷眾多的更新目前已經更新到最新的3.5版本,開發(fā)環(huán)境中自帶的標準外設庫為2.0.3版本,本書中以比較穩(wěn)定而且較新的V3.4版本為基礎介紹標準外設庫的結構。
可以從ST的官方網站下載到各種版本的標準外設庫,首先看一下3.4版本標準外設庫的文件結構,如圖 5?3所示。3.0以上版本的文件結構大致相同,每個版本可能略有調整。
圖 5?3 STM32F10XXX V3.4標準外設庫文件結構
表 5?4中介紹了每個文件夾所包含的主要內容。
表 5?4 STM32F10XXX V3.4標準外設庫文件夾描述
STM32F10x_StdPeriph_Lib_V3.4.0 | _htmresc | 包含了所有的html頁面資源 |
---|---|---|
Li aries |
CMSIS | 見表 5?6 |
STM32F10x_StdPeriph_Driver | inc | 標準外設庫驅動頭文件 |
src | 標準外設庫驅動源文件 | ? |
Project | Examples | 標準外設庫驅動的完整例程 |
Template | MDK-ARM | KEIL RVMDK的項目模板示例 |
RIDE | Raisonance RIDE的項目模板示例 | ? |
EWARM | IAR EWARM的項目模板示例 | ? |
Utilities | STM3210-EVAL | 包含了用于STM3210B-EVAL和STM3210E-EVAL評估板的專用驅動 |
標準外設庫的第一部分是CMSIS 和STM32F10x_StdPeriph_Driver,CMSIS是獨立于供應商的Cortex-M處理器系列硬件抽象層,為芯片廠商和中間件供應商提供了簡單的處理器軟件接口,簡化了軟件復用工作,降低了Cortex-M 操作系統(tǒng)的移植難度,并減少了新入門的微控制器開發(fā)者的學習曲線和新產品的上市時間。
STM32F10x_StdPeriph_Driver則包括了分別對應包括了所有外設對應驅動函數,這些驅動函數均使用C語言編寫,并提供了統(tǒng)一的易于調用的函數接口,供開發(fā)者使用。
Project文件夾中則包括了ST官方的所有例程和基于不同編譯器的項目模板,這些例程是學習和使用STM32的重要參考。Utilities包含了相關評估板的示例程序和驅動函數,供使用官方評估板的開發(fā)者使用,很多驅動函數同樣可以作為學習的重要參考。
STM32F10xxx標準外設庫體系結構如圖 5?4所示。圖中很好的展示了各層以及具體文件之間的聯(lián)系,各文件的具體功能說明如表 5?5所示。
圖 5?4 STM32F10xxx標準外設庫體系結構
表 5?5 文件功能說明
文件名 | 功能描述 | 具體功能說明 |
---|---|---|
core_cm3.hcore_cm3.c | Cortex-M3內核及其設備文件 | 訪問Cortex-M3內核及其設備:NVIC,SysTick等訪問Cortex-M3的CPU寄存器和內核外設的函數 |
stm32f10x.h | 微控制器專用頭文件 | 這個文件包含了STM32F10x全系列所有外設寄存器的定義(寄存器的基地址和布局)、位定義、中斷向量表、存儲空間的地址映射等 |
system_stm32f10x.hsystem_stm32f10x.c | 微控制器專用系統(tǒng)文件 | 函數SystemInit,用來初始化微控制器函數Sysem_ExtMemCtl,用來配置外部存儲器控制器。它位于文件startup_stm32f10x_xx.s /.c,在跳轉到main前調用SystemFrequncy,該值代表系統(tǒng)時鐘頻率 |
startup_stm32f10x_Xd.s | 編譯器啟動代碼 | 微控制器專用的中斷處理程序列表(與頭文件一致)弱定義(Weak)的中斷處理程序默認函數(可以被用戶代碼覆蓋) 該文件是與編譯器相關的 |
stm32f10x_conf.h | 固件庫配置文件 | 通過更改包含的外設頭文件來選擇固件庫所使用的外設,在新建程序和進行功能變更之前應當首先修改對應的配置。 |
stm32f10x_it.hstm32f10x_it.c | 外設中斷函數文件 | 用戶可以相應的加入自己的中斷程序的代碼,對于指向同一個中斷向量的多個不同中斷請求,用戶可以通過判斷外設的中斷標志位來確定準確的中斷源,執(zhí)行相應的中斷服務函數。 |
stm32f10x_ppp.hstm32f10x_ppp.c | 外設驅動函數文件 | 包括了相關外設的初始化配置和部分功能應用函數,這部分是進行編程功能實現(xiàn)的重要組成部分。 |
Application.c | 用戶文件 | 用戶程序文件,通過標準外設庫提供的接口進行相應的外設配置和功能設計。 |
基于CMSIS標準的軟件架構
根據調查研究,軟件開發(fā)已經被嵌入式行業(yè)公認為最主要的開發(fā)成本。
對于ARM公司來說,一個ARM內核往往會授權給多個廠家,生產種類繁多的產品,如果沒有一個通用的軟件接口標準,那么當開發(fā)者在使用不同廠家的芯片時將極大的增加了軟件開發(fā)成本,因此,ARM與Atmel、IAR、Keil、hami-nary Micro、Micrium、NXP、SEGGER和ST等諸多芯片和軟件廠商合作,將所有Cortex芯片廠商產品的軟件接口標準化,制定了CMSIS標準。
此舉意在降低軟件開發(fā)成本,尤其針對新設備項目開發(fā),或者將已有軟件移植到其他芯片廠商提供的基于Cortex處理器的微控制器的情況。有了該標準,芯片廠商就能夠將他們的資源專注于產品外設特性的差異化,并且消除對微控制器進行編程時需要維持的不同的、互相不兼容的標準的需求,從而達到降低開發(fā)成本的目的。如圖5?5所示,?基于CMSIS標準的軟件架構主要分為以下4層:用戶應用層、操作系統(tǒng)及中間件接口層、CMSIS層、硬件寄存器層。?其中CMSIS層起著承上啟下的作用:一方面該層對硬件寄存器層進行統(tǒng)一實現(xiàn),屏蔽了不同廠商對Cortex-M系列微處理器核內外設寄存器的不同定義;另一方面又向上層的操作系統(tǒng)及中間件接口層和應用層提供接口,簡化了應用程序開發(fā)難度,使開發(fā)人員能夠在完全透明的情況下進行應用程序開發(fā)。也正是如此,CMSIS層的實現(xiàn)相對復雜。
圖 5?5 CMSIS標準的軟件架構
層主要分為以下3 個部分:
(1) 核內外設訪問層(CPAL,Core Peripheral Access Layer):該層由ARM 負責實現(xiàn)。包括對寄存器名稱、地址的定義,對核寄存器、NVIC、調試子系統(tǒng)的訪問接口定義以及對特殊用途寄存器的訪問接口(例如:CONTROL,xPSR)定義。由于對特殊寄存器的訪問以內聯(lián)方式定義,所以針對不同的編譯器ARM 統(tǒng)一用來屏蔽差異。該層定義的接口函數均是可重入的。
(2) 片上外設訪問層(DPAL, Device Peripheral Access Layer):該層由芯片廠商負責實現(xiàn)。該層的實現(xiàn)與CPAL 類似,負責對硬件寄存器地址以及外設訪問接口進行定義。該層可調用CPAL 層提供的接口函數同時根據設備特性對異常向量表進行擴展,以處理相應外設的中斷請求。
(3) 外設訪問函數(AFP, Access Functions for Peripherals):該層也由芯片廠商負責實現(xiàn),主要是提供訪問片上外設的訪問函數,這一部分是可選的。
對一個Cortex-M微控制系統(tǒng)而言,CMSIS通過以上三個部分實現(xiàn)了:
定義了訪問外設寄存器和異常向量的通用方法;
定義了核內外設的寄存器名稱和核異常向量的名稱;
為RTOS 核定義了與設備獨立的接口,包括Debug通道。
這樣芯片廠商就能專注于對其產品的外設特性進行差異化,并且消除他們對微控制器進行編程時需要維持的不同的、互相不兼容的標準需求,以達到低成本開發(fā)的目的。CMSIS中的具體文件結構如表 5?6所示。
表 5?6 CMSIS文件夾結構
CMSIS | Core | Documentation | CMSIS文檔 | ? |
---|---|---|---|---|
CM3 | Startup | arm | MDK ARM編譯器啟動文件 | startup_stm32f10x_hd.s: 大容量產品啟動文件startup_stm32f10x_md.s: 中容量產品啟動文件startup_stm32f10x_ld.s: 小容量產品啟動文件 |
gcc_ride7 | GCC編譯器啟動文件 | ? | ? | ? |
iar | IAR編譯器啟動文件 | ? | ? | ? |
TrueSTUDIO | TrueSTUDIO編譯器啟動文件 | ? | ? | ? |
本文件夾包含STMF10xxx CMSIS文件:微控制器外設訪問層和內核設備訪問層:core_cm3.h:CMSIS的Cortex-M3內核設備訪問層頭文件core_cm3.c:CMSIS的Cortex-M3內核設備訪問層源文件stm32f10x.h:CMSIS的Cortex-M3 STM32f10xxx微控制器外設訪問層頭文件system_stm32f10x.h:CMSIS的Cortex-M3 STM32f10xxx微控制器外設訪問層頭文件system_stm32f10x.c:CMSIS的Cortex-M3 STM32f10xxx微控制器外設訪問層源文件 | ? | ? | ? | ? |
在實際開發(fā)過程中,根據應用程序的需要,可以采取2種方法使用標準外設庫(StdPeriph_Lib):
(1) 使用外設驅動:這時應用程序開發(fā)基于外設驅動的API(應用編程接口)。用戶只需要配置文件”stm32f10x_conf.h”,并使用相應的文件”stm32f10x_ppp.h/.c”即可。
(2) 不使用外設驅動:這時應用程序開發(fā)基于外設的寄存器結構和位定義文件。
這兩種方法的優(yōu)缺點在“使用標準外設庫開發(fā)的優(yōu)勢”小節(jié)中已經有了具體的介紹,這里仍要說明的是,使用使用標準外設庫進行開發(fā)可以極大的減小軟件開發(fā)的工作量,也是目前嵌入式系統(tǒng)開發(fā)的一個趨勢。
標準外設庫(StdPeriph_Lib)支持STM32F10xxx系列全部成員:大容量,中容量和小容量產品。?從表 5?6中也可以看出,啟動文件已經對不同的系列進行了劃分,實際開發(fā)中根據使用的STM32產品具體型號,用戶可以通過文件”stm32f10x.h”中的預處理define或者通過開發(fā)環(huán)境中的全局設置來配置標準外設庫(StdPeriph_Lib),一個define對應一個產品系列。
下面列出支持的產品系列
STM32F10x_LD:STM32小容量產品
STM32F10x_MD:STM32中容量產品
STM32F10x_HD:STM32大容量產品
在庫文件中這些define的具體作用范圍是:
文件“stm3210f.h”中的中斷IRQ定義
啟動文件中的向量表,小容量,中容量,大容量產品各有一個啟動文件
外設存儲器映像和寄存器物理地址
產品設置:外部晶振(HSE)的值等
系統(tǒng)配置函數
因此通過宏定義這種方式,可以使標準外設庫適用于不同系列的產品,同時也方便與不同產品之間的軟件移植,極大的方便了軟件的開發(fā)。
1.1.4 STM32F10XXX標準外設庫的使用
標準外設庫中包含了眾多的變量定義和功能函數,如果不能了解他們的命名規(guī)范和使用規(guī)律將會給編程帶來很大的麻煩,本節(jié)將主要敘述標準外設庫中的相關規(guī)范,通過這些規(guī)范的學習可以更加靈活的使用固件庫,同時也將極大增強程序的規(guī)范性和易讀性,同時標準外設庫中的這種規(guī)范也值得我們在進行其他相關的開發(fā)時使用和借鑒。
縮寫定義
標準外設庫中的主要外設均采用了縮寫的形式,通過這些縮寫可以很容易的辨認對應的外設。
縮寫 | 外設/單元 |
---|---|
ADC | 模數轉換器 |
BKP | 備份寄存器 |
CAN | 控制器局域網模塊 |
CEC | ? |
CRC | CRC計算單元 |
DAC | 數模轉換器 |
DBGMCU | 調試支持 |
DMA | 直接內存存取控制器 |
EXTI | 外部中斷事件控制器 |
FLASH | 閃存存儲器 |
FSMC | 靈活的靜態(tài)存儲器控制器 |
GPIO | 通用輸入輸出 |
I2C | I2C接口 |
IWDG | 獨立看門狗 |
PWR | 電源/功耗控制 |
RCC | 復位與時鐘控制器 |
RTC | 實時時鐘 |
SDIO | SDIO接口 |
SPI | 串行外設接口 |
TIM | 定時器 |
USART | 通用同步/異步收發(fā)器 |
WWDG | 窗口看門狗 |
命名規(guī)則
標準外設庫遵從以下命名規(guī)則 PPP表示任一外設縮寫,例如:ADC。源程序文件和頭文件命名都以“stm32f10x_”作為開頭,例如:stm32f10x_conf.h。常量僅被應用于一個文件的,定義于該文件中;被應用于多個文件的,在對應頭文件中定義。所有常量都由英文字母大寫書寫。
寄存器作為常量處理。他們的命名都由英文字母大寫書寫。在大多數情況下,他們采用與縮寫規(guī)范一致。外設函數的命名以該外設的縮寫加下劃線為開頭。每個單詞的第一個字母都由英文字母大寫書寫,例如:SPI_SendData。
在函數名中,只允許存在一個下劃線,用以分隔外設縮寫和函數名的其它部分。對于函數命名,總的來說有以下規(guī)則:
名為PPP_Init的函數,其功能是根據PPP_InitTypeDef中指定的參數,初始化外設PPP,例如TIM_Init.
名為PPP_DeInit的函數,其功能為復位外設PPP的所有寄存器至缺省值,例如TIM_DeInit.
名為PPP_Init的函數,其功能為通過設置PPP_InitTypeDef 結構中的各種參數來定義外設的功能,例如:USART_Init .
名為PPP_Cmd的函數,其功能為使能或者失能外設PPP,例如:SPI_Cmd.
名為PPP_ITConfig的函數,其功能為使能或者失能來自外設PPP某中斷源,例如:RCC_ITConfig.
名為PPP_DMAConfig的函數,其功能為使能或者失能外設PPP的DMA接口,例如:TIM1_DMAConfig.
用以配置外設功能的函數,總是以字符串“Config”結尾,例如GPIO_PinRemapConfig.
名為PPP_GetFlagStatus的函數,其功能為檢查外設PPP某標志位被設置與否,例如:I2C_GetFlagStatus.
名為PPP_ClearFlag的函數,其功能為清除外設PPP標志位,例如:I2C_ClearFlag.
名為PPP_GetITStatus的函數,其功能為判斷來自外設PPP的中斷發(fā)生與否,例如:I2C_GetITStatus.
名為PPP_ClearITPendingBit的函數,其功能為清除外設PPP中斷待處理標志位,例如:I2C_ClearITPendingBit.
這樣的命名方式非常便于程序的編寫和閱讀,以標準外設庫中的示例函數為例,下面摘錄了STM32F10x_StdPeriph_ExamplesADCADCs_DMAmian.c中的一段程序。
?
DMA_InitType Def DMA_InitStructure; /*?DMA1?channel1?configuration?*/ DMA_DeInit(DMA1_Channel1); DMA_InitStructure.DMA_PeripheralBaseAddr?=?ADC1_DR_Address; DMA_InitStructure.DMA_MemoryBaseAddr?=?(uint32_t)&ADC1ConvertedValue; DMA_InitStructure.DMA_DIR?=?DMA_DIR_PeripheralSRC; DMA_InitStructure.DMA_BufferSize?=?1; DMA_InitStructure.DMA_PeripheralInc?=?DMA_PeripheralInc_Disable; DMA_InitStructure.DMA_MemoryInc?=?DMA_MemoryInc_Disable; DMA_InitStructure.DMA_PeripheralDataSize?=?DMA_PeripheralDataSize_HalfWord; DMA_InitStructure.DMA_MemoryDataSize?=?DMA_MemoryDataSize_HalfWord; DMA_InitStructure.DMA_Mode?=?DMA_Mode_Circular; DMA_InitStructure.DMA_Priority = DMA_Priority_High; DMA_InitStructure.DMA_M2M?=?DMA_M2M_Disable; DMA_Init(DMA1_Channel1,?&DMA_InitStructure); /* Enable DMA1 channel1 */ DMA_Cmd(DMA1_Channel1,?ENABLE);??
?
這段程序完成了DMA1通道的配置,首先定義了DMA_InitType DMA_InitStructure,接著配置DMA_InitType的各種參數,各參數的命名方式也均使用約定的命名方式,從命名就能夠很容易的看出各參數所指代的具體功能。
功能參數配置完成后,使用DMA_Init(DMA1_Channel1, &DMA_InitStructure);完成相應外設的初始化,最后使用DMA_Cmd(DMA1_Channel1, ENABLE) 使能相應外設。從這個例子就能夠很容易的看出標準外設庫這種規(guī)范化的命名規(guī)則給編寫和閱讀程序帶來的好處。
3. 變量定義
在早期的版本中有24個變量定義,在Keil的安裝根目錄下,可以找到對應的定義,路徑為:KeilARMINCSTSTM32F10xstm32f10x_type.h
?
/*?Includes?-------*/ /*?Exported?types?---------*/ typedef?signed?long?s32; typedef?signed?short?s16; typedef?signed?char?s8; typedef signed long const sc32; /* Read Only */ typedef signed short const sc16; /* Read Only */ typedef signed char const sc8; /* Read Only */ typedef volatile signed long vs32; typedef volatile signed short vs16; typedef volatile signed char vs8; typedef volatile signed long const vsc32; /* Read Only */ typedef volatile signed short const vsc16; /* Read Only */ typedef volatile signed char const vsc8; /* Read Only */ typedef unsigned long u32; 30 31 typedef unsigned short u16; typedef unsigned char u8; typedef unsigned long const uc32; /* Read Only */ typedef unsigned short const uc16; /* Read Only */ typedef unsigned char const uc8; /* Read Only */ typedef volatile unsigned long vu32; typedef volatile unsigned short vu16; typedef volatile unsigned char vu8; typedef volatile unsigned long const vuc32; /* Read Only */ typedef volatile unsigned short const vuc16; /* Read Only */ typedef volatile unsigned char const vuc8; /* Read Only */?
3.0以后的版本中使用了CMSIS數據類型,變量的定義有所不同,但是出于兼容舊版本的目的,以上的數據類型仍然兼容。
CMSIS的IO類型限定詞如表 5?7所示,CMSIS和STM32固件庫的數據類型對比如表 5?8所示。這些數據類型可以在: STM32F10x_StdPeriph_Lib_V3.4.0LibrariesCMSISCM3DeviceSupportSTSTM32F10xstm32f10x.h中找到具體的定義,此部分定義如下。 ?
/*!?
表 5?7 CMSIS IO類型限定詞
IO類限定詞 | #define | 描述 |
---|---|---|
_I | volatile const | 只讀訪問 |
_O | volatile | 只寫訪問 |
_IO | volatile | 讀和寫訪問 |
表 5?8 固件庫與CMSIS數據類型對比
固件庫類型 | CMSIS類型 | 描述 |
---|---|---|
s32 | int32_t | 易揮發(fā)只讀有符號32位數據 |
s16 | int16_t | 易揮發(fā)只讀有符號16位數據 |
s8 | int8_t | 易揮發(fā)只讀有符號8位數據 |
sc32 | const int32_t | 只讀有符號32位數據 |
sc16 | const int16_t | 只讀有符號16位數據 |
sc8 | const int8_t | 只讀有符號8位數據 |
vs32 | _IO int32_t | 易揮發(fā)讀寫訪問有符號32位數據 |
vs16 | _IO int16_t | 易揮發(fā)讀寫訪問有符號16位數據 |
vs8 | _IO int8_t | 易揮發(fā)讀寫訪問有符號8位數據 |
vsc32 | _I int32_t | 易揮發(fā)只讀有符號32位數據 |
vsc16 | _I int16_t | 易揮發(fā)只讀有符號16位數據 |
vsc8 | _I int8_t | 易揮發(fā)只讀有符號8位數據 |
u32 | uint32_t | 無符號32位數據 |
u16 | uint16_t | 無符號16位數據 |
u8 | uint8_t | 無符號8位數據 |
uc32 | const uint32_t | 只讀無符號32位數據 |
uc16 | const uint16_t | 只讀無符號16位數據 |
uc8 | const uint8_t | 只讀無符號8位數據 |
vu32 | _IO uint32_t | 易揮發(fā)讀寫訪問無符號32位數據 |
vu16 | _IO uint16_t | 易揮發(fā)讀寫訪問無符號16位數據 |
vu8 | _IO uint8_t | 易揮發(fā)讀寫訪問無符號8位數據 |
vuc32 | _I uint32_t | 易揮發(fā)只讀無符號32位數據 |
vuc16 | _I uint16_t | 易揮發(fā)只讀無符號16位數據 |
vuc8 | _I uint8_t | 易揮發(fā)只讀無符號8位數據 |
stm32f10x.h文件中還包含了常用的布爾形變量定義,如:
?
typedef enum {RESET = 0, SET = !RESET} FlagStatus, ITStatus; typedef enum {DISABLE = 0, ENABLE = !DISABLE} FunctionalState; #define IS_FUNCTIONAL_STATE(STATE) (((STATE) == DISABLE) || ((STATE) == ENABLE)) typedef enum {ERROR = 0, SUCCESS = !ERROR} ErrorStatus;?
?
不同版本的標準外設庫的變量定義略有不同,如3.4版本中就沒有之前版本的TRUE和FALSE的定義,用戶也可以根據自己的需求按照上面的格式定義自己的布爾形變量。
在使用標準外設庫進行開發(fā)遇到相關的定義問題時應首先找到對應的頭文件定義。
使用步驟
如果在開發(fā)中使用標準外設庫需要哪些描述呢?下面就進行簡要的介紹,這兒介紹的使用方法是與開發(fā)環(huán)境無關的,在不同的開發(fā)環(huán)境中可能在操作方式上略有不同,但是總體的流程都是一樣的,下一小節(jié)將介紹在MDK ARM開發(fā)環(huán)境下使用標準外設庫的詳細過程。
首先新建一個項目并設置工具鏈對應的啟動文件,可以使用標準外設庫中提供的模板,也可以自己根據自己的需求新建。標準外設庫中已經提供了不同工具鏈對應的文件,位于:
STM32F10x_StdPeriph_Lib_V3.4.0LibrariesCMSISCM3DeviceSupportSTSTM32F10xstartup目錄下。
其次按照使用產品的具體型號選擇具體的啟動文件,加入工程。文件主要按照使用產品的容量進行區(qū)分,根據產品容量進行選擇即可。每個文件的具體含義可以在“stm32f10x.h”文件中找到對應的說明,摘錄如下:
?
#if?!defined?(STM32F10X_LD)?&&?!defined?(STM32F10X_LD_VL)? && !defined (STM32F10X_MD) && !defined (STM32F10X_MD_VL) && !defined (STM32F10X_HD) && !defined (STM32F10X_HD_VL) && !defined (STM32F10X_XL) && !defined (STM32F10X_CL) /*?#define?STM32F10X_LD?*/? /*!< STM32F10X_LD: STM32 Low density devices */ /*?#define?STM32F10X_LD_VL?*/? /*!< STM32F10X_LD_VL: STM32 Low density Value Line devices */ /*?#define?STM32F10X_MD?*/? /*!< STM32F10X_MD: STM32 Medium density devices */ /*?#define?STM32F10X_MD_VL?*/? /*!< STM32F10X_MD_VL: STM32 Medium density Value Line devices */ /* #define STM32F10X_HD */ /*!< STM32F10X_HD: STM32 High density devices */ /*?#define?STM32F10X_HD_VL?*/? /*!< STM32F10X_HD_VL: STM32 High density value line devices */ /*?#define?STM32F10X_XL?*/? /*!< STM32F10X_XL: STM32 XL-density devices */ /* #define STM32F10X_CL */ /*!< STM32F10X_CL: STM32 Connectivity line devices */ #endif /* Tip: To avoid modifying this file each time you need to switch between these devices,?you?can?define?the?device?in?your?toolchain?compiler?preprocessor.? - Low-density devices are STM32F101xx, STM32F102xx and STM32F103xx microcontrollers where the Flash memory density ranges between 16 and 32 Kbytes. - Low-density value line devices are STM32F100xx microcontrollers where the Flash memory density ranges between 16 and 32 Kbytes. - Medium-density devices are STM32F101xx, STM32F102xx and STM32F103xx microcontrollers where the Flash memory density ranges between 64 and 128 Kbytes. - Medium-density value line devices are STM32F100xx microcontrollers where the Flash memory density ranges between 64 and 128 Kbytes. - High-density devices are STM32F101xx and STM32F103xx microcontrollers where the Flash memory density ranges between 256 and 512 Kbytes. - High-density value line devices are STM32F100xx microcontrollers where the Flash memory density ranges between 256 and 512 Kbytes. - XL-density devices are STM32F101xx and STM32F103xx microcontrollers where the Flash memory density ranges between 512 and 1024 Kbytes. - Connectivity line devices are STM32F105xx and STM32F107xx microcontrollers. */
?
“stm32f10x.h”是整個標準外設庫的入口文件,這個文件包含了STM32F10x全系列所有外設寄存器的定義(寄存器的基地址和布局)、位定義、中斷向量表、存儲空間的地址映射等。為了是這個文件適用于不同系列的產品,程序中是通過宏定義來實現(xiàn)不同產品的匹配的,上面這段程序的注釋中已經詳細給出了每個啟動文件所對應的產品系列,與之對應,也要相應的修改這個入口文件,需要根據所使用的產品系列正確的注釋/去掉相應的注釋define。
在這段程序的下方同樣有這樣的一個注釋程序:
/#define USE_STDPERIPH_DRIVER/
用于選擇是否使用標準外設庫,如果保留這個注釋,則用戶開發(fā)程序可以基于直接訪問“stm32f10x.h”中定義的外設寄存器,所有的操作均基于寄存器完成,目前不使用固件庫的單片機開發(fā),如51、AVR、MSP430等其實都是采用此種方式,通過在對應型號的頭文件中進行外設寄存器等方面的定義,從而在程序中對相應的寄存器操作完成相應的功能設計。
如果去掉/#define USE_STDPERIPH_DRIVER/的注釋,則是使用標準外設庫進行開發(fā),用戶需要使用在文件“stm32f10x_conf.h”中,選擇要用的外設,外設同樣是通過注釋/去掉注釋的方式來選擇。示例程序如下:
?
/* Uncomment the line below to enable peripheral header file inclusion */ #include?"stm32f10x_adc.h" /* #include "stm32f10x_bkp.h" */ /* #include "stm32f10x_can.h" */ /* #include "stm32f10x_cec.h" */ /* #include "stm32f10x_crc.h" */ /* #include "stm32f10x_dac.h" */ /*?#include?"stm32f10x_dbgmcu.h"?*/ #include?"stm32f10x_dma.h" /* #include "stm32f10x_exti.h" */ /* #include "stm32f10x_flash.h" */ /* #include "stm32f10x_fsmc.h" */ #include "stm32f10x_gpio.h" /* #include "stm32f10x_i2c.h" */ /* #include "stm32f10x_iwdg.h" */ /* #include "stm32f10x_pwr.h" */ #include "stm32f10x_rcc.h" /* #include "stm32f10x_rtc.h" */ /* #include "stm32f10x_sdio.h" */ /* #include "stm32f10x_spi.h" */ /* #include "stm32f10x_tim.h" */ /* #include "stm32f10x_usart.h" */ /* #include "stm32f10x_wwdg.h" */ #include "misc.h" /* High level functions for NVIC and SysTick (add-on to CMSIS functions) */? 上面一段程序來自于例程中的AD采集程序,程序使用了AD和DMA,因此去掉相應的注釋,同時幾乎所有的應用都需要使用復位與時鐘以及通用I/O,因此這兩項是必須的,而多數程序同樣要使用NVIC中斷IRQ設置和SysTick時鐘源設置,那么 “misc.h”這一項也是必須的。
上面已經針對具體的產品信號和程序功能進行了針對性的配置,接下來需要配置系統(tǒng)所使用的時鐘,系統(tǒng)時鐘在“system_stm32f10x.c”同樣通過注釋的方式來配置,程序如下:? ?
#if defined (STM32F10X_LD_VL) || (defined STM32F10X_MD_VL) || (defined STM32F10X_HD_VL) /* #define SYSCLK_FREQ_HSE HSE_VALUE */ #define SYSCLK_FREQ_24MHz 24000000 #else /* #define SYSCLK_FREQ_HSE HSE_VALUE */ /* #define SYSCLK_FREQ_24MHz 24000000 */ /* #define SYSCLK_FREQ_36MHz 36000000 */ /* #define SYSCLK_FREQ_48MHz 48000000 */ /* #define SYSCLK_FREQ_56MHz 56000000 */ #define SYSCLK_FREQ_72MHz 72000000 #endif? 如果這兒沒有明確的定義那么HSI時鐘將會作為系統(tǒng)時鐘。
至此,已經配置了系統(tǒng)的主要外部參數,這些參數主要是通過更改相關的宏定義來實現(xiàn)的,有些開發(fā)環(huán)境,例如Keil支持在軟件設置中加入全局宏定義,因此像芯片系列定義,是否使用固件庫定義等也可以通過軟件添加來實現(xiàn)。
完成了主要參數配置以后就可以進行程序的開發(fā)了,標準外設庫開發(fā)就可以使用標準外設庫中提供的方便的API函數進行相應的功能設計了。配置完成后,程序中仍然可以直接更改相應寄存器的配置,通過對寄存器的操作可以提高程序的效率,因此可以使用標準外設庫和寄存器操作兩種相結合的方式。
?
審核編輯:湯梓紅
?
評論
查看更多