GD32F303 驅動 W25Q64


F303 驅動 W25Q64 遇到的問題(收的的數據被右移一位)

 

問題出現:

  昨天在移植W25Q64的驅動到F3飛控上,使用正點原子的F4的驅動,細節處做了修改,但是無論如何器件ID無法正確讀出,一直讀出 FFEF,但手冊上是說明 EF16。

定位問題:

  首先看到讀取芯片ID的函數

復制代碼
//讀取芯片ID
u16 SPI_Flash_ReadID(void)
{
    u16 Temp = 0;
    
    SPI_FLASH_CS_L();
    SPI2_ReadWriteByte(0X90);   //發送讀取ID命令
    SPI2_ReadWriteByte(0x00);
    SPI2_ReadWriteByte(0x00);
    SPI2_ReadWriteByte(0x00);
    ( SPI2_ReadWriteByte(0x00); ) 
    Temp|=SPI2_ReadWriteByte(0xFF)<<8;
    Temp|=SPI2_ReadWriteByte(0xFF);
    SPI_FLASH_CS_H();
    return Temp;
}
復制代碼

 

  其中 三句 SPI2_ReadWriteByte(0x00);是發送3個任意幀,然后第四幀就可以開始讀取數據,於是,我就多加了一句(括號中),發現讀出ef16,這說明,STM32的硬件SPI配置無誤,SPI可以通訊,但是數據像是被右移了一位。但是程序此處不能修改,問題應該出現在SPI收發,因為此時W25Q64依舊無法讀取數據,不可能在每一個W25Q64要讀取前多加這一條語句,這無法根本解決問題。

  於是看到SPI收發函數

復制代碼
//SPIx 讀寫一個字節
//TxData:要寫入的字節
//返回值:讀取到的字節
u8 SPI2_ReadWriteByte(u8 TxData)
{        
//    u8 retry=0;  
    
    while (SPI_I2S_GetFlagStatus(SPI2, SPI_I2S_FLAG_TXE) == RESET) //檢查指定的SPI標志位設置與否:發送緩存空標志位
    {
//        retry++;
//        if(retry>200)return 0;
    }
    SPI_SendData8(SPI2, TxData); //通過外設SPIx發送一個數據
//    SPI_I2S_SendData16(SPI2, TxData);
//    retry=0;

    while (SPI_I2S_GetFlagStatus(SPI2, SPI_I2S_FLAG_RXNE) == RESET)//檢查指定的SPI標志位設置與否:接受緩存非空標志位
    {
//        retry++;
//        if(retry>200)return 0;
    }                                  
    return SPI_ReceiveData8(SPI2); //返回通過SPIx最近接收的數據 
//    return SPI_I2S_ReceiveData16(SPI2);
}
復制代碼

  其中,未打注釋就是原子的例程,后來我查到很多人用 SPI_I2S_ReceiveData();這個函數,區別就是這句函數形參是u16,而例程是u8,會不會是這里的問題呢?於是我用打注釋的兩句發送/接收函數代替了例程的發送/接收函數,結果仍是不對.....

  整的一天都在上網查資料,修改,失敗....不斷循環

  今天早上一來,靈光一現。難道F303和F4有什么細微不同我還沒發現?(我這么想是因為之前移植軟件IIC的驅動,F3無響應,而F1有響應,查了快一周才發現F3是不支持位帶操作的),於是再檢查SPI初始化函數(昨天看了不知道多少遍了...),配合網上一查,看到ST官方F3 Discover板子的SPI初始化,在使能SPI前,還有一句函數 SPI_RxFIFOThresholdConfig(SPI2, SPI_RxFIFOThreshold_QF); 於是加上去試試,沒想到正常了,ID讀取正確,可對Flsah執行讀取/存入操作!問題完美解決!

下面是SPI初始化函數(不包含CS腳)

 

復制代碼
void SPI2_Init(void)
{
    GPIO_InitTypeDef GPIO_InitStructure;
    SPI_InitTypeDef  SPI_InitStructure;
    
    RCC_AHBPeriphClockCmd( SPI2_GPIO_RCC, ENABLE );
    RCC_APB1PeriphClockCmd(SPI2_RCC, ENABLE);

    GPIO_InitStructure.GPIO_Mode  = GPIO_Mode_AF; 
    GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;
    GPIO_InitStructure.GPIO_Pin   = SPI2_CLK_Pin|SPI2_DI_Pin|SPI2_DO_Pin;
    GPIO_InitStructure.GPIO_PuPd  = GPIO_PuPd_UP;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_Level_3;
    GPIO_Init(SPI2_GPIO_Port, &GPIO_InitStructure);
    
    GPIO_PinAFConfig(SPI2_GPIO_Port, SPI2_CLK_PinSource, GPIO_AF_5);
    GPIO_PinAFConfig(SPI2_GPIO_Port, SPI2_DI_PinSource,  GPIO_AF_5);
    GPIO_PinAFConfig(SPI2_GPIO_Port, SPI2_DO_PinSource,  GPIO_AF_5);

    SPI_InitStructure.SPI_Direction         = SPI_Direction_2Lines_FullDuplex;  //設置SPI單向或者雙向的數據模式:SPI設置為雙線雙向全雙工
    SPI_InitStructure.SPI_Mode              = SPI_Mode_Master;      //設置SPI工作模式:設置為主SPI
    SPI_InitStructure.SPI_DataSize          = SPI_DataSize_8b;      //設置SPI的數據大小:SPI發送接收8位幀結構
    SPI_InitStructure.SPI_CPOL              = SPI_CPOL_High;        //串行同步時鍾的空閑狀態為高電平
    SPI_InitStructure.SPI_CPHA              = SPI_CPHA_2Edge;       //串行同步時鍾的第二個跳變沿(上升或下降)數據被采樣
    SPI_InitStructure.SPI_NSS               = SPI_NSS_Soft;         //NSS信號由硬件(NSS管腳)還是軟件(使用SSI位)管理:內部NSS信號有SSI位控制
    SPI_InitStructure.SPI_BaudRatePrescaler = SPI_BaudRatePrescaler_256;    //定義波特率預分頻的值:波特率預分頻值為256
    SPI_InitStructure.SPI_FirstBit          = SPI_FirstBit_MSB;     //指定數據傳輸從MSB位還是LSB位開始:數據傳輸從MSB位開始
    SPI_InitStructure.SPI_CRCPolynomial     = 7;                    //CRC值計算的多項式
    SPI_Init(SPI2, &SPI_InitStructure);  
 
    SPI_RxFIFOThresholdConfig(SPI2, SPI_RxFIFOThreshold_QF);
    
    SPI_Cmd(SPI2, ENABLE); //使能SPI外設
}
復制代碼

 

 

 

原因分析:

  官方對這個函數的說明是 "Configures the FIFO reception threshold for the selected SPI." 也就是對選中的SPI的接收FIFO的閾值進行配置。SPI_RxFIFOThreshold_QF: RXNE event is generated if the FIFO level is greater or equal to 1/4.也就是 如果FIFO使用大於等於1/4RXNE事件才發生。


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM