SPI協(xié)議是由摩托羅拉公司提出的通訊協(xié)議(Serial Peripheral Interface),即串行外圍設備接口,是一種高速全雙工的通信總線。它在芯片的管腳上只占用四根線,節(jié)約了芯片的管腳,同時為PCB在布局上節(jié)省了空間,提供方便,正是出于這種簡單易用的特性,現(xiàn)在越來越多的芯片集成了這種通信協(xié)議,它被廣泛地使用在ADC、LCD、FLASH等設備與MCU之間的通信。
CKS32F4xx系列產品SPI介紹
CKS32F4xx系列的SPI外設可用作通訊的主機及從機,支持最高的SCK時鐘頻率為fpclk/2(CKS32F407型號的芯片默認fpclk為142MHz,fpclk2為84MHz),完全支持SPI協(xié)議的4種模式。SPI協(xié)議根據CPOL及CPHA的不同狀態(tài)分成的四種工作模式如下表所示:

CKS32F4xx系列的SPI架構如下圖所示:

圖中的1處是SPI的引腳MOSI、MISO、SCK、NSS。CKS32F4xx芯片有多個 SPI外設,它們的SPI通訊信號引出到不同GPIO引腳上,使用時必須配置到這些指定的引腳。關于GPIO引腳的復用功能可以查閱芯片數(shù)據手冊。各個引腳的作用介紹如下:
(1)NSS:從設備選擇信號線,常稱為片選信號線。當有多個SPI從設備與 SPI主機相連時,設備的其它信號線SCK、MOSI及MISO同時并聯(lián)到相同的SPI 總線上,即無論有多少個從設備,都共同只使用這3條總線;而每個從設備都有獨立的一條NSS信號線,當主機要選擇從設備時,把該從設備的NSS信號線設置為低電平,該從設備即被選中,即片選有效,接著主機開始與被選中的從設備進行SPI通訊。所以SPI通訊以NSS線置低電平為開始信號,以NSS線被拉高作為結束信號。
(2)SCK:時鐘信號線,用于通訊數(shù)據同步。它由通訊主機產生,決定了通訊的速率,不同的設備支持的最高時鐘頻率不一樣,兩個設備之間通訊時,通訊速率受限于低速設備。
(3)MOSI:主設備輸出/從設備輸入引腳。主機的數(shù)據從這條信號線輸出,從機由這條信號線讀入主機發(fā)送的數(shù)據,即這條線上數(shù)據的方向為主機到從機。
(4)MISO:主設備輸入/從設備輸出引腳。主機從這條信號線讀入數(shù)據,從機的數(shù)據由這條信號線輸出到主機,即在這條線上數(shù)據的方向為從機到主機。
圖中的2處是SCK線的時鐘信號,由波特率發(fā)生器根據“控制寄存器CR1”中的BR[0:2]位控制,該位是對fpclk時鐘的分頻因子,對fpclk的分頻結果就是SCK引腳的輸出時鐘頻率。
圖中的3處是SPI的數(shù)據控制邏輯。SPI的MOSI及MISO都連接到數(shù)據移位寄存器上,數(shù)據移位寄存器的內容來源于接收緩沖區(qū)及發(fā)送緩沖區(qū)以及MISO、MOSI線。當向外發(fā)送數(shù)據的時候,數(shù)據移位寄存器以“發(fā)送緩沖區(qū)”為數(shù)據源,把數(shù)據一位一位地通過數(shù)據線發(fā)送出去;當從外部接收數(shù)據的時候,數(shù)據移位寄存器把數(shù)據線采樣到的數(shù)據一位一位地存儲到“接收緩沖區(qū)”中。通過寫SPI 的“數(shù)據寄存器DR”把數(shù)據填充到發(fā)送緩沖區(qū)中,通過“數(shù)據寄存器DR”,可以獲取接收緩沖區(qū)中的內容。其中數(shù)據幀的長度可以通過“控制寄存器CR1”的“DFF位”配置成8位及16位模式;配置“LSBFIRST位”可選擇MSB先行還是 LSB先行。
圖中的4處是SPI的整體控制邏輯。整體控制邏輯負責協(xié)調整個SPI外設,控制邏輯的工作模式根據我們配置的“控制寄存器(CR1/CR2)”的參數(shù)而改變,基本的控制參數(shù)包括SPI模式、波特率、LSB先行、主從模式、單雙向模式等等。在外設工作時,控制邏輯會根據外設的工作狀態(tài)修改“狀態(tài)寄存器(SR)”,我們只要讀取狀態(tài)寄存器相關的寄存器位,就可以了解SPI的工作狀態(tài)了。除此之外,控制邏輯還根據要求,負責控制產生SPI中斷信號、DMA請求及控制NSS信號線。實際應用中,我們一般不使用CKS32 SPI外設的標準NSS信號線,而是更簡單地使用普通的GPIO,軟件控制它的電平輸出,從而產生通訊起始和停止信號。
CKS32F4xx系列的SPI作為通訊主機端時收發(fā)數(shù)據的過程如下:
(1) 控制NSS信號線,產生起始信號;
(2) 把要發(fā)送的數(shù)據寫入到“數(shù)據寄存器DR”中,該數(shù)據會被存儲到發(fā)送緩沖區(qū);
(3) 通訊開始,SCK時鐘開始運行。MOSI把發(fā)送緩沖區(qū)中的數(shù)據一位一位地傳輸出去;MISO則把數(shù)據一位一位地存儲進接收緩沖區(qū)中;
(4) 當發(fā)送完一幀數(shù)據的時候,“狀態(tài)寄存器SR”中的“TXE標志位”會被置1,表示傳輸完一幀,發(fā)送緩沖區(qū)已空;類似地,當接收完一幀數(shù)據的時候,“RXNE標志位”會被置1,表示傳輸完一幀,接收緩沖區(qū)非空;
(5) 等待到“TXE標志位”為1時,若還要繼續(xù)發(fā)送數(shù)據,則再次往“數(shù)據寄存器DR”寫入數(shù)據即可;等待到“RXNE標志位”為1時,通過讀取“數(shù)據寄存器DR”可以獲取接收緩沖區(qū)中的內容。
假如我們使能了TXE或RXNE中斷,TXE或RXNE置1時會產生SPI中斷信號,進入同一個中斷服務函數(shù),到SPI中斷服務程序后,可通過檢查寄存器位來了解是哪一個事件,再分別進行處理。也可以使用DMA方式來收發(fā)“數(shù)據寄存器 DR”中的數(shù)據。
CKS32F4xx系列產品SPI的配置
接下來我們講解如何利用CKS32F4xx系列固件庫來完成對SPI的配置使用。跟其它外設一樣,CKS32標準庫提供了SPI初始化結構體及初始化函數(shù)來配置 SPI外設。了解初始化結構體后我們就能對SPI外設運用自如了,代碼如下:
typedef struct
{
uint16_t SPI_Direction;
uint16_t SPI_Mode;
uint16_t SPI_DataSize;
uint16_t SPI_CPOL;
uint16_t SPI_CPHA;
uint16_t SPI_NSS;
uint16_t SPI_BaudRatePrescaler;
uint16_t SPI_FirstBit;
uint16_t SPI_CRCPolynomial;
}SPI_InitTypeDef;
結構體中各個成員變量的介紹及初始化時可被賦的值如下:
1) SPI_Direction:本成員設置SPI的通訊方向,可設置為雙線全雙工 (SPI_Direction_2Lines_FullDuplex),雙線只接收 (SPI_Direction_2Lines_RxOnly),單線只接收(SPI_Direction_1Line_Rx)、單線只發(fā)送模式(SPI_Direction_1Line_Tx)。
2) SPI_Mode:本成員設置SPI工作在主機模式(SPI_Mode_Master)或從機模式(SPI_Mode_Slave ),這兩個模式的最大區(qū)別為SPI的SCK信號線的時序,SCK的時序是由通訊中的主機產生的。若被配置為從機模式,CKS32的SPI外設將接受外來的SCK信號:
3) SPI_DataSize: 本成員可以選擇SPI通訊的數(shù)據幀大小是為8位 (SPI_DataSize_8b)還是16位(SPI_DataSize_16b)。
4) SPI_CPOL和SPI_CPHA: 這兩個成員配置SPI的時鐘極性CPOL和時鐘相位CPHA,前面講過這兩個配置影響到SPI的通訊模式。時鐘極性CPOL成員可設置為高電平(SPI_CPOL_High)或低電平(SPI_CPOL_Low )。時鐘相位CPHA則 可以設置為SPI_CPHA_1Edge(在SCK的奇數(shù)邊沿采集數(shù)據)或 SPI_CPHA_2Edge(在SCK的偶數(shù)邊沿采集數(shù)據)。
5) SPI_NSS: 本成員配置NSS引腳的使用模式,可以選擇為硬件模式 (SPI_NSS_Hard )與軟件模式(SPI_NSS_Soft ),在硬件模式中的SPI片選信號由 SPI硬件自動產生,而軟件模式則需要我們自己把相應的GPIO端口拉高或置低產生非片選和片選信號。實際中軟件模式應用比較多。
6) SPI_BaudRatePrescaler: 本成員設置波特率分頻因子,分頻后的時鐘即為SPI的SCK信號線的時鐘頻率。這個成員參數(shù)可設置為fpclk的2、4、6、8、16、32、64、128、256分頻。可選的值如下所示:
SPI_BaudRatePrescaler_2 //2分頻 SPI_BaudRatePrescaler_4 //4分頻 SPI_BaudRatePrescaler_6 //6分頻 SPI_BaudRatePrescaler_8 //8分頻 SPI_BaudRatePrescaler_16 //16分頻 SPI_BaudRatePrescaler_32 //32分頻 SPI_BaudRatePrescaler_64 //64分頻 SPI_BaudRatePrescaler_128 //128分頻 SPI_BaudRatePrescaler_256 //256分頻?
7) SPI_FirstBit:所有串行的通訊協(xié)議都會有MSB先行(高位數(shù)據在前)還是 LSB先行(低位數(shù)據在前)的問題,而CKS32F4xx系列的SPI模塊可以通過這個結構體成員,對這個特性編程控制。
SPI_FirstBit_MSB //高位數(shù)據在前 SPI_FirstBit_LSB //低位數(shù)據在前
8) SPI_CRCPolynomial:這是SPI的CRC校驗中的多項式,若我們使用CRC 校驗時,就使用這個成員的參數(shù)(多項式),來計算CRC的值。
配置完這些結構體成員的值,調用庫函數(shù)SPI_Init即可把結構體的配置寫入到寄存器中。
CKS32F4xx讀寫SPI FLASH實驗
串口的DMA接發(fā)通信實驗是存儲器到外設和外設到存儲器的數(shù)據傳輸。在第24
本小節(jié)以一種使用SPI通訊的串行FLASH存儲芯片的讀寫實驗為大家講解 CKS32F4xx系列的SPI使用方法。實驗中的FLASH芯片(型號:W25Q32)是一種使用SPI通訊協(xié)議的NORFLASH存儲器,它的CS/CLK/DIO/DO引腳分別連接到了CKS32F4xx對應的SPI引腳NSS/SCK/MOSI/MISO上,其中CKS32F4xx的NSS引腳是一個普通的GPIO,不是SPI的專用NSS引腳,所以程序中我們要使用軟件控制的方式。
1.編程要點
(1) 初始化通訊使用的目標引腳及端口時鐘;
(2) 使能SPI外設的時鐘;
(3) 配置SPI外設的模式、地址、速率等參數(shù)并使能SPI外設;
(4) 編寫基本SPI按字節(jié)收發(fā)的函數(shù);
(5) 編寫對FLASH擦除及讀寫操作的的函數(shù);
(6) 編寫測試程序,對讀寫數(shù)據進行校驗。
2.代碼分析
代碼清單1:W25Q32初始化配置
void W25QXX_Init(void)
{
GPIO_InitTypeDef GPIO_InitStructure;
RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOD, ENABLE);
RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOG, ENABLE);
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_OUT;
GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_100MHz;
GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP;
GPIO_Init(GPIOD, &GPIO_InitStructure);
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_3;
GPIO_Init(GPIOG, &GPIO_InitStructure);
GPIO_SetBits(GPIOG,GPIO_Pin_3);
W25QXX_CS=1; //SPI FLASH不選中
SPI1_Init(); //初始化SPI
SPI1_SetSpeed(SPI_BaudRatePrescaler_4); //設置為21M時鐘
W25QXX_TYPE=W25QXX_ReadID(); //讀取FLASH ID.
}
上面的代碼主要是完成對W25Q32片選引腳的初始化,SPI初始化。SPI通信速率設置和讀取W25Q32的ID。
代碼清單2:SPI初始化函數(shù)
void SPI1_Init(void)
{
GPIO_InitTypeDef GPIO_InitStructure;
SPI_InitTypeDef SPI_InitStructure;
RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOB, ENABLE);
RCC_APB2PeriphClockCmd(RCC_APB2Periph_SPI1, ENABLE);
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_3|GPIO_Pin_4|GPIO_Pin_5;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;
GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_100MHz;
GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP;
GPIO_Init(GPIOB, &GPIO_InitStructure);
GPIO_PinAFConfig(GPIOB,GPIO_PinSource3,GPIO_AF_SPI1);
GPIO_PinAFConfig(GPIOB,GPIO_PinSource4,GPIO_AF_SPI1);
GPIO_PinAFConfig(GPIOB,GPIO_PinSource5,GPIO_AF_SPI1);
RCC_APB2PeriphResetCmd(RCC_APB2Periph_SPI1,ENABLE);
RCC_APB2PeriphResetCmd(RCC_APB2Periph_SPI1,DISABLE);
SPI_InitStructure.SPI_Direction = SPI_Direction_2Lines_FullDuplex;
SPI_InitStructure.SPI_Mode = SPI_Mode_Master;
SPI_InitStructure.SPI_DataSize = SPI_DataSize_8b;
SPI_InitStructure.SPI_CPOL = SPI_CPOL_High;
SPI_InitStructure.SPI_CPHA = SPI_CPHA_2Edge;
SPI_InitStructure.SPI_NSS = SPI_NSS_Soft;
SPI_InitStructure.SPI_BaudRatePrescaler = SPI_BaudRatePrescaler_256;
SPI_InitStructure.SPI_FirstBit = SPI_FirstBit_MSB;
SPI_InitStructure.SPI_CRCPolynomial = 7;
SPI_Init(SPI1, &SPI_InitStructure);
SPI_Cmd(SPI1, ENABLE);
SPI1_ReadWriteByte(0xff);
}
上面這段代碼主要是完成對SPI1的初始化,首先是配置了SPI1使用的引腳SPI1_SCK、SPI1_MOSI和SPI1_MISO。然后是根據第2小節(jié)的內容完成對SPI1外設模式的配置。根據FLASH芯片W25Q32的說明,它支持SPI模式0及模式3,支持雙線全雙工,使用MSB先行模式,支持最高通訊時鐘為104MHz,數(shù)據幀長度為8位。我們要把CKS32F4的SPI外設中的這些參數(shù)配置一致。
代碼清單3:SPI1單字節(jié)收發(fā)函數(shù)
u8 SPI1_ReadWriteByte(u8 TxData)
{
while (SPI_I2S_GetFlagStatus(SPI1, SPI_I2S_FLAG_TXE) == RESET){}
SPI_I2S_SendData(SPI1, TxData);
while (SPI_I2S_GetFlagStatus(SPI1, SPI_I2S_FLAG_RXNE) == RESET){}
return SPI_I2S_ReceiveData(SPI1);
}
本函數(shù)中不包含SPI起始和停止信號,只是收發(fā)的主要過程,所以在調用本函數(shù)前后要做好起始和停止信號的操作。通過檢測TXE標志,獲取發(fā)送緩沖區(qū)的狀態(tài),若發(fā)送緩沖區(qū)為空,則表示可能存在的上一個數(shù)據已經發(fā)送完畢;等待至發(fā)送緩沖區(qū)為空后,調用庫函數(shù)SPI_I2S_SendData把要發(fā)送的數(shù)據“TxData”寫入到SPI的數(shù)據寄存器DR,寫入SPI數(shù)據寄存器的數(shù)據會存儲到發(fā)送緩沖區(qū),由SPI外設發(fā)送出去;寫入完畢后等待RXNE事件,即接收緩沖區(qū)非空事件。由于SPI雙線全雙工模式下MOSI與MISO數(shù)據傳輸是同步的,當接收緩沖區(qū)非空時,表示上面的數(shù)據發(fā)送完畢,且接收緩沖區(qū)也收到新的數(shù)據;等待至接收緩沖區(qū)非空時,通過調用庫函數(shù)SPI_I2S_ReceiveData讀取SPI的數(shù)據寄存器DR,就可以獲取接收緩沖區(qū)中的新數(shù)據了。代碼中使用關鍵字“return”把接收到的這個數(shù)據作為SPI1_ReadWriteByte函數(shù)的返回值。
搞定了SPI的基本收發(fā)單元后,還需要了解如何對FLASH芯片進行讀寫。FLASH 芯片自定義了很多指令,我們通過控制CKS32F4利用SPI總線向FLASH 芯片發(fā)送指令,F(xiàn)LASH芯片收到后就會執(zhí)行相應的操作。具體的指令代碼可以查看W25Q32芯片的數(shù)據手冊。
代碼清單4:讀取FLASH芯片ID函數(shù)
u16 W25QXX_ReadID(void)
{
u16 Temp = 0;
W25QXX_CS=0;
SPI1_ReadWriteByte(0x90);
SPI1_ReadWriteByte(0x00);
SPI1_ReadWriteByte(0x00);
SPI1_ReadWriteByte(0x00);
Temp|=SPI1_ReadWriteByte(0xFF)<<8;
Temp|=SPI1_ReadWriteByte(0xFF);
W25QXX_CS=1;
return Temp;
}
這段代碼利用控制CS引腳電平的宏“W25QXX_CS”以及前面編寫的單字節(jié)收發(fā)函數(shù)SPI1_ReadWriteByte,很清晰地實現(xiàn)了讀ID指令的時序,最后把讀 取到的這3個數(shù)據合并到一個變量Temp中,然后作為函數(shù)返回值,把該返回值與我們定義的芯片ID對比,即可知道FLASH芯片是否正常。
代碼清單5:W25Q32寫使能和寫禁止函數(shù)
void W25QXX_Write_Enable(void)
{
W25QXX_CS=0;
SPI1_ReadWriteByte(W25X_WriteEnable);
W25QXX_CS=1;
}
void W25QXX_Write_Disable(void)
{
W25QXX_CS=0;
SPI1_ReadWriteByte(W25X_WriteDisable);
W25QXX_CS=1;
}
由于FLASH存儲器的特性決定了它只能把原來為“1”的數(shù)據位改寫成“0”,而原來為“0”的數(shù)據位不能直接改寫為“1”。所以在寫入前,必須要對目標存儲矩陣進行擦除操作,把矩陣中的數(shù)據位擦除為“1”,在數(shù)據寫入的時候,如果要存儲數(shù)據“1”, 那就不修改存儲矩陣,在要存儲數(shù)據“0”時,需要更改該位。W25Q32支持“扇區(qū)擦除”、“塊擦除”以及“整片擦除”。扇區(qū)擦除指令的第一個字節(jié)為指令編碼,緊接著發(fā)送的3個字節(jié)用于表示要擦除的存儲矩陣地址。要注意的是在扇區(qū)擦除指令前,還需要先發(fā)送“寫使能”指令,發(fā)送扇區(qū)擦除指令后,通過讀取寄存器狀態(tài)等待扇區(qū)擦除操作完畢。
代碼清單6:W25Q32扇區(qū)擦除函數(shù)
void W25QXX_Erase_Sector(u32 Dst_Addr)
{
Dst_Addr*=4096;
W25QXX_Write_Enable();
W25QXX_Wait_Busy();
W25QXX_CS=0;
SPI1_ReadWriteByte(W25X_SectorErase);
SPI1_ReadWriteByte((u8)((Dst_Addr)>>16));
SPI1_ReadWriteByte((u8)((Dst_Addr)>>8));
SPI1_ReadWriteByte((u8)Dst_Addr);
W25QXX_CS=1;
W25QXX_Wait_Busy();
}
目標扇區(qū)被擦除完畢后,就可以向它寫入數(shù)據了。與EEPROM類似,F(xiàn)LASH芯片也有頁寫入命令,使用頁寫入命令最多可以一次向FLASH傳輸256個字節(jié)的數(shù)據,我們把這個單位稱為頁大小。在進行頁寫入時第1個字節(jié)為“頁寫入指令”編碼,2-4字節(jié)為要寫入的“地址A”,接著的是要寫入的內容,最多可以發(fā)送 256字節(jié)數(shù)據,這些數(shù)據將會從“地址A”開始,按順序寫入到FLASH的存儲矩陣。若發(fā)送的數(shù)據超出256個,則會覆蓋前面發(fā)送的數(shù)據。
代碼清單7:W25Q32頁寫入函數(shù)
void W25QXX_Write_Page(u8* pBuffer,u32 WriteAddr,u16 NumByteToWrite)
{
u16 i;
W25QXX_Write_Enable();
W25QXX_CS=0;
SPI1_ReadWriteByte(W25X_PageProgram);
SPI1_ReadWriteByte((u8)((WriteAddr)>>16));
SPI1_ReadWriteByte((u8)((WriteAddr)>>8));
SPI1_ReadWriteByte((u8)WriteAddr);
for(i=0;i
應用的時候我們常常要寫入不定量的數(shù)據,直接調用“頁寫入”函數(shù)并不是特別方便,所以我們頁寫入函數(shù)的基礎上編寫了“不定量數(shù)據寫入”的函數(shù)。在實際調用這個“不定量數(shù)據寫入”函數(shù)時,還要注意確保目標扇區(qū)處于擦除狀態(tài)
代碼清單8:W25Q32不定量數(shù)據寫入函數(shù)
void W25QXX_Write(u8* pBuffer,u32 WriteAddr,u16 NumByteToWrite)
{
u32 secpos;
u16 secoff;
u16 secremain;
u16 i;
u8 * W25QXX_BUF;
W25QXX_BUF=W25QXX_BUFFER;
secpos=WriteAddr/4096;
secoff=WriteAddr%4096;
secremain=4096-secoff;
if(NumByteToWrite<=secremain)secremain=NumByteToWrite;
while(1)
{
W25QXX_Read(W25QXX_BUF,secpos*4096,4096);
for(i=0;i4096)secremain=4096;
else secremain=NumByteToWrite;
}
};
}
函數(shù)的入口參數(shù)pBuffer是數(shù)據存儲區(qū)、WriteAd是開始寫入的地址(24bit)、NumByteToWrite是要寫入的字節(jié)數(shù)(最大65535)gaojp。
相對于寫入,F(xiàn)LASH芯片W25Q32的數(shù)據讀取要簡單的多,發(fā)送了指令編碼及要讀的起始地址和要讀取的字節(jié)數(shù)之后,F(xiàn)LASH 芯片W25Q32就會按地址遞增的方式返回存儲矩陣中一定字節(jié)數(shù)量的數(shù)據。
代碼清單9:W25Q32讀取數(shù)據函數(shù)
void W25QXX_Read(u8* pBuffer,u32 ReadAddr,u16 NumByteToRead)
{
u16 i;
W25QXX_CS=0;
SPI1_ReadWriteByte(W25X_ReadData);
SPI1_ReadWriteByte((u8)((ReadAddr)>>16));
SPI1_ReadWriteByte((u8)((ReadAddr)>>8));
SPI1_ReadWriteByte((u8)ReadAddr);
for(i=0;i
函數(shù)的入口參數(shù)pBuffer是數(shù)據存儲區(qū)、ReadAddr是開始讀取的地址(24bit)、NumByteToRead是要讀取的字節(jié)數(shù)(最大65535)。
完成基本的讀寫函數(shù)后,接下來我們編寫一個讀寫測試函數(shù)來檢驗驅動程。
代碼清單10:W25Q32讀寫測試函數(shù)
uint8_t w25q32_Test(void)
{
u16 i;
printf("寫入的數(shù)據:
");
for ( i=0; i<=10; i++ )
{
spi_Buf_Write[i] = i;
printf("0x%02X ", spi_Buf_Write[i]);
}
W25QXX_Write((u8*)spi_Buf_Write,FLASH_SIZE-100,11);
printf("寫成功,");
printf("讀出的數(shù)據:
");
W25QXX_Read(datatemp,FLASH_SIZE-100,11);
for (i=0; i<11; i++)
{
if(datatemp[i] != spi_Buf_Write[i])
{
printf("0x%02X ", datatemp[i]);
printf("錯誤:I2C EEPROM寫入與讀出的數(shù)據不一致");
return 0;
}
printf("0x%02X ", datatemp[i]);
}
printf("
");
printf("spi(w25q32)讀寫測試成功");
return 1;
}
代碼中先填充一個數(shù)組,數(shù)組的內容為0,1至10,接著把這個數(shù)組的內容寫入到SPI FLASH中,并將寫入的數(shù)據打印輸出到串口調試助手。寫入完畢后再從SPI FLASH的地址中讀取數(shù)據,把讀取到的數(shù)據與寫入的數(shù)據進行校驗,若一致說明讀寫正常,否則讀寫過程有問題或者SPI FLASH芯片不正常,然后再將讀取到的數(shù)據打印輸出到串口調試助手。
代碼清單11:主函數(shù)
int main(void)
{
u16 id = 0;
NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);
delay_init(168);
USART_Configuration();
W25QXX_Init();
while(1)
{
id = W25QXX_ReadID();
if (id == W25Q32 || id == NM25Q32)
break;
printf("W25Q32 init failed
");
delay_ms(500);
delay_ms(500);
}
printf("W25Q32 init success
");
w25q32_Test();
while(1)
{
}
}
主函數(shù)代碼比較簡單,主要是完成串口初始化和W25Q32的初始化,初始化完成之后會執(zhí)行W25QXX_ReadID函數(shù),讀取W25Q32的ID,同時對ID進行判斷,并將結果通過串口調試助手打印輸出。然后會執(zhí)行一次W25Q32測試函數(shù),并將一些測試結果通過串口調試助手打印輸出。
審核編輯:湯梓紅
-
mcu
+關注
關注
147文章
18429瀏覽量
380799 -
接口
+關注
關注
33文章
9308瀏覽量
155718 -
通信
+關注
關注
18文章
6269瀏覽量
139274 -
SPI
+關注
關注
17文章
1840瀏覽量
99083 -
通訊協(xié)議
+關注
關注
10文章
293瀏覽量
21223
原文標題:MCU微課堂 | CKS32F4xx系列產品SPI通信
文章出處:【微信號:中科芯MCU,微信公眾號:中科芯MCU】歡迎添加關注!文章轉載請注明出處。
發(fā)布評論請先 登錄
MCU微課堂|CKS32F4xx系列產品時鐘配置
CKS32F4xx系列產品NVIC中斷優(yōu)先級管理單元講解
CKS32F4xx系列產品串口DMA傳輸
CKS32F4xx系列產品的定時器使用-基本特征和定時操作
CKS32F4xx系列RNG功能設置
CKS32F4xx系列FSMC功能簡介

CKS32F4xx系列產品SPI通信
評論