STM32微控制器SPI通信功能配置詳解
在嵌入式系統(tǒng)開(kāi)發(fā)中,串行通信協(xié)議是連接微控制器與外圍設(shè)備的核心技術(shù)。SPI(Serial Peripheral Interface)作為一種高速、全雙工、同步的串行通信總線,因其簡(jiǎn)單高效的特點(diǎn),被廣泛應(yīng)用于傳感器、存儲(chǔ)器、顯示屏等設(shè)備的連接。本文以STM32系列微控制器為例,系統(tǒng)闡述SPI通信的原理、配置方法及實(shí)踐應(yīng)用,幫助開(kāi)發(fā)者快速掌握這一關(guān)鍵技術(shù)。
一、SPI通信協(xié)議基礎(chǔ)
1.1 SPI協(xié)議的核心特性
SPI協(xié)議采用主從架構(gòu)設(shè)計(jì),支持一主多從的通信模式。其核心優(yōu)勢(shì)包括:
?全雙工通信?:支持同時(shí)發(fā)送和接收數(shù)據(jù),提高傳輸效率。
?同步時(shí)鐘?:通過(guò)主設(shè)備產(chǎn)生的時(shí)鐘信號(hào)實(shí)現(xiàn)數(shù)據(jù)同步,避免時(shí)序混亂。
?靈活配置?:支持多種數(shù)據(jù)幀格式(如8位、16位)和時(shí)鐘頻率。
?硬件支持?:STM32的硬件SPI模塊可減輕CPU負(fù)擔(dān),提升系統(tǒng)性能。
1.2 SPI信號(hào)線定義
SPI總線由4根信號(hào)線構(gòu)成:
?SCK(Serial Clock)?:時(shí)鐘信號(hào)線,由主設(shè)備產(chǎn)生,控制數(shù)據(jù)傳輸時(shí)序。
?MOSI(Master Out Slave In)?:主設(shè)備數(shù)據(jù)輸出,從設(shè)備數(shù)據(jù)輸入。
?MISO(Master In Slave Out)?:主設(shè)備數(shù)據(jù)輸入,從設(shè)備數(shù)據(jù)輸出。
?NSS(Slave Select)?:從設(shè)備選擇信號(hào),低電平有效,用于選擇特定從設(shè)備。
1.3 SPI通信模式
SPI協(xié)議通過(guò)時(shí)鐘極性和相位(CPOL/CPHA)組合定義4種工作模式:
?模式0?:CPOL=0(時(shí)鐘空閑低電平),CPHA=0(數(shù)據(jù)采樣于第1個(gè)邊沿)。
?模式1?:CPOL=0,CPHA=1(數(shù)據(jù)采樣于第2個(gè)邊沿)。
?模式2?:CPOL=1(時(shí)鐘空閑高電平),CPHA=0。
?模式3?:CPOL=1,CPHA=1。
二、STM32硬件SPI模塊特性
2.1 STM32的SPI外設(shè)優(yōu)勢(shì)
STM32系列微控制器集成硬件SPI模塊,具有以下特點(diǎn):
?多通道支持?:部分型號(hào)提供3個(gè)獨(dú)立SPI接口(如SPI1/SPI2/SPI3)。
?時(shí)鐘分頻?:支持1/2~1/256的分頻系數(shù),適應(yīng)不同速度需求。
?數(shù)據(jù)幀格式?:支持8/16位數(shù)據(jù)長(zhǎng)度,MSB/LSB優(yōu)先傳輸。
?中斷/DMA支持?:可配置中斷或DMA傳輸,降低CPU占用率。
2.2 關(guān)鍵寄存器配置
STM32的SPI模塊通過(guò)以下寄存器實(shí)現(xiàn)功能控制:
?CR1(控制寄存器1)?:配置CPOL/CPHA、數(shù)據(jù)長(zhǎng)度、時(shí)鐘分頻等。
?CR2(控制寄存器2)?:設(shè)置NSS管理方式(硬件/軟件控制)。
?DR(數(shù)據(jù)寄存器)?:用于數(shù)據(jù)讀寫(xiě)操作。
?SR(狀態(tài)寄存器)?:提供傳輸完成標(biāo)志(TXE/RXNE)。
三、SPI通信配置步驟
3.1 引腳配置
使用STM32CubeMX工具可快速完成引腳分配:
選擇SPI接口(如SPI1)。
配置SCK、MOSI、MISO為復(fù)用推挽輸出(AF_PP)。
設(shè)置NSS引腳為推挽輸出(GPIO_Mode_Out_PP)。
3.2 參數(shù)初始化
通過(guò)HAL庫(kù)函數(shù)配置SPI參數(shù):
cCopy CodeSPI_HandleTypeDef hspi1;
hspi1.Instance = SPI1;
hspi1.Init.Mode = SPI_MODE_MASTER;
hspi1.Init.Direction = SPI_DIRECTION_2LINES;
hspi1.Init.DataSize = SPI_DATASIZE_8BIT;
hspi1.Init.CLKPolarity = SPI_POLARITY_LOW; // CPOL=0
hspi1.Init.CLKPhase = SPI_PHASE_1EDGE; // CPHA=0
hspi1.Init.NSS = SPI_NSS_SOFT;
hspi1.Init.BaudRatePrescaler = SPI_BAUDRATEPRESCALER_16;
hspi1.Init.FirstBit = SPI_FIRSTBIT_MSB;
hspi1.Init.TIMode = SPI_TIMODE_DISABLE;
HAL_SPI_Init(&hspi1);
3.3 數(shù)據(jù)傳輸實(shí)現(xiàn)
3.3.1 輪詢方式
cCopy Codeuint8_t tx_data = 0x55;
uint8_t rx_data;
HAL_SPI_TransmitReceive(&hspi1, &tx_data, &rx_data, 1, HAL_MAX_DELAY);
3.3.2 中斷方式
cCopy CodeHAL_SPI_TransmitReceive_IT(&hspi1, &tx_data, &rx_data, 1);
// 在中斷回調(diào)函數(shù)中處理接收數(shù)據(jù)
void HAL_SPI_TxRxCpltCallback(SPI_HandleTypeDef *hspi) {
if (hspi->Instance == SPI1) {
// 處理接收數(shù)據(jù)
}
}
3.3.3 DMA方式
cCopy CodeHAL_SPI_TransmitReceive_DMA(&hspi1, &tx_data, &rx_data, 1);
// 在DMA完成回調(diào)中處理數(shù)據(jù)
void HAL_SPI_TxRxCpltCallback(SPI_HandleTypeDef *hspi) {
if (hspi->Instance == SPI1) {
// 處理接收數(shù)據(jù)
}
}
四、SPI通信實(shí)踐案例
4.1 連接W25Q128 Flash存儲(chǔ)器
配置步驟:
初始化SPI1(模式0,8MHz時(shí)鐘)。
發(fā)送Flash指令序列:
寫(xiě)使能:0x06
頁(yè)編程:0x02 + 地址 + 數(shù)據(jù)
讀數(shù)據(jù):0x03 + 地址
實(shí)現(xiàn)數(shù)據(jù)讀寫(xiě)函數(shù):
cCopy Codevoid SPI_Flash_Write(uint32_t addr, uint8_t *data, uint32_t len) {
HAL_SPI_Transmit(&hspi1, (uint8_t*)&addr, 4, 100);
HAL_SPI_Transmit(&hspi1, data, len, 100);
}
void SPI_Flash_Read(uint32_t addr, uint8_t *data, uint32_t len) {
HAL_SPI_Transmit(&hspi1, (uint8_t*)&addr, 4, 100);
HAL_SPI_Receive(&hspi1, data, len, 100);
}
4.2 連接OLED顯示屏
配置要點(diǎn):
設(shè)置SPI為模式3(CPOL=1, CPHA=1)。
實(shí)現(xiàn)顯示函數(shù):
cCopy Codevoid OLED_Write_Command(uint8_t cmd) {
uint8_t data = {0x80 | (cmd & 0x3F), 0x00};
HAL_SPI_Transmit(&hspi1, data, 2, 10);
}
void OLED_Write_Data(uint8_t data) {
uint8_t cmd = {0x40 | (data & 0x3F), 0x00};
HAL_SPI_Transmit(&hspi1, cmd, 2, 10);
}
五、SPI通信常見(jiàn)問(wèn)題及解決方案
5.1 通信失敗問(wèn)題
?現(xiàn)象?:數(shù)據(jù)傳輸錯(cuò)誤或超時(shí)。
?排查步驟?:
檢查硬件連接(SCK/MOSI/MISO/NSS)。
驗(yàn)證時(shí)鐘頻率是否超過(guò)從設(shè)備支持范圍。
確認(rèn)CPOL/CPHA模式與從設(shè)備匹配。
檢查NSS信號(hào)是否正常拉低。
5.2 性能優(yōu)化技巧
?DMA傳輸?:對(duì)于大數(shù)據(jù)量傳輸,使用DMA可顯著提升效率。
?時(shí)鐘分頻?:根據(jù)實(shí)際需求調(diào)整時(shí)鐘頻率,避免過(guò)高導(dǎo)致信號(hào)失真。
?中斷優(yōu)先級(jí)?:合理設(shè)置SPI中斷優(yōu)先級(jí),避免與其他外設(shè)沖突。
5.3 多從機(jī)系統(tǒng)設(shè)計(jì)
?硬件NSS?:每個(gè)從設(shè)備連接獨(dú)立的NSS引腳,由主設(shè)備控制。
?軟件NSS?:通過(guò)GPIO模擬NSS信號(hào),需手動(dòng)控制選擇時(shí)序。
?菊花鏈拓?fù)?:多個(gè)從設(shè)備串聯(lián),但需支持菊花鏈模式的芯片。
STM32的硬件SPI模塊為開(kāi)發(fā)者提供了高效可靠的通信解決方案。通過(guò)合理配置參數(shù)、選擇適當(dāng)?shù)膫鬏敺绞?,可滿足絕大多數(shù)外設(shè)連接需求。隨著物聯(lián)網(wǎng)技術(shù)的發(fā)展,SPI協(xié)議在低功耗、實(shí)時(shí)性方面的優(yōu)勢(shì)將進(jìn)一步凸顯。未來(lái),SPI可能會(huì)與新興技術(shù)如PCIe協(xié)議相結(jié)合,在保持簡(jiǎn)單性的同時(shí)提升傳輸性能。





