OpenEdv-开源电子网

 找回密码
 立即注册
正点原子全套STM32/Linux/FPGA开发资料,上千讲STM32视频教程免费下载...
查看: 3148|回复: 7

关于SPI读取FLASH ID的问题

[复制链接]

6

主题

20

帖子

0

精华

初级会员

Rank: 2

积分
74
金钱
74
注册时间
2019-5-24
在线时间
16 小时
发表于 2020-4-26 17:29:48 | 显示全部楼层 |阅读模式
1金钱
STM32F407的板子,想用串口打印读取的FLASH的ID,一个是用的90H指令,一个是ABH,但是打印出来的ID一个是0xffff,一个是0xff,请问哪里有问题啊

#include "spi_flash.h"
#include "sys.h"
#include "usart.h"

/*SPI初始化函数*/
void Spi_Flash_Init(void)
{
        GPIO_InitTypeDef  GPIO_InitStructure;
        SPI_InitTypeDef SPI_InitStruct;
       
        RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOB, ENABLE);
        RCC_APB2PeriphClockCmd(RCC_APB2Periph_SPI1,ENABLE);
        RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOG, ENABLE);//使能GPIOG时钟
       
        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_InitStructure.GPIO_Pin = GPIO_Pin_14;                //片选信号
  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_InitStructure.GPIO_Pin = GPIO_Pin_7;//PG7
  GPIO_Init(GPIOG, &GPIO_InitStructure);//初始化
        GPIO_SetBits(GPIOG,GPIO_Pin_7);//PG7输出1,防止NRF干扰SPI FLASH的通信

        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);//复位SPI1
        RCC_APB2PeriphResetCmd(RCC_APB2Periph_SPI1,DISABLE);//停止复位SPI1
       
        SPI_InitStruct.SPI_Mode = SPI_Mode_Master;
        SPI_InitStruct.SPI_BaudRatePrescaler = SPI_BaudRatePrescaler_2;
        SPI_InitStruct.SPI_CPOL = SPI_CPOL_High;
        SPI_InitStruct.SPI_CPHA = SPI_CPHA_2Edge;
        SPI_InitStruct.SPI_DataSize = SPI_DataSize_8b;
        SPI_InitStruct.SPI_Direction = SPI_Direction_2Lines_FullDuplex;
        SPI_InitStruct.SPI_FirstBit = SPI_FirstBit_MSB;
        SPI_InitStruct.SPI_NSS = SPI_NSS_Soft;
        SPI_InitStruct.SPI_CRCPolynomial = 0;
       
        SPI_Init(SPI1,&SPI_InitStruct);
        SPI_Cmd(SPI1,ENABLE);
}
//可以用来检查等待是否超时,并发送错误代号方便发现错误
//uint16_t checkerror(u8 code)
//{
//        printf("SPI等待等待超时! errorcode=%d\n",code);
//        return 0;
//}

/*        发送数据函数  由于是全双工工作模式,发送一个数据后会自动接收一个数据
        writebyte为要发送的8位数据
        rebyte        为接收到的数据
*/
u8 WritReadeByte(u8 writebyte)                                                               
{
        uint8_t rebyte;
        //等待TXE位为1,才能够写入数据
        while(SPI_I2S_GetFlagStatus(SPI1,SPI_I2S_FLAG_TXE) == RESET)
        {       
        }
        SPI_I2S_SendData(SPI1,writebyte);
        while(SPI_I2S_GetFlagStatus(SPI1,SPI_I2S_FLAG_RXNE) == RESET)
        {
        }       
        rebyte=SPI_I2S_ReceiveData(SPI1);
        return rebyte;
}


u8 Spi_Flash_ReadID(void)
{
        u8 id;
        //FLASH的读取ID指令
        Spi_Flash_Cs = 1;
        Spi_Flash_Cs=0;                //将SPI的片选信号拉低才能进行数据传送
        WritReadeByte(0xAB);
        WritReadeByte(DUMMY);
        WritReadeByte(DUMMY);
        WritReadeByte(DUMMY);       
        //接收读取到的ID
        id=WritReadeByte(DUMMY);
        Spi_Flash_Cs = 1;        //发送完一个数据将片选拉低
        return id;
       
}

u16 W25QXX_ReadID(void)
{
        u16 Temp = 0;          
        Spi_Flash_Cs=0;                                    
        WritReadeByte(0x90);//发送读取ID命令            
        WritReadeByte(0x00);             
        WritReadeByte(0x00);             
        WritReadeByte(0x00);                                    
        Temp|=WritReadeByte(0xFF)<<8;  
        Temp|=WritReadeByte(0xFF);         
        Spi_Flash_Cs=1;                                    
        return Temp;
}                       

void write_enable()
{
        Spi_Flash_Cs=0;
        WritReadeByte(0x06);
        Spi_Flash_Cs=1;
}       

#include "stm32f4xx.h"
#include "sys.h"
#include "spi_flash.h"
#include "usart.h"
#include "led.h"
#include "delay.h"
int main(void)
{
        u8 i;
        u8 FLASHID;
        u16 flash_id ;
        delay_init(168);
        uart_init(115200);
        LED_Init();
        Spi_Flash_Init();
        LED0=0;
        //printf("FLASF ID is 0x");
        //flash_id= W25QXX_ReadID();
         write_enable();
        FLASHID=Spi_Flash_ReadID();
        printf("FLASF ID2 is 0x%x\r\n",FLASHID);
        flash_id=W25QXX_ReadID();
        printf("FLASF ID1 is 0x%x\r\n",flash_id);
        for(i=3;i>0;i--)
        {
        //printf("FLASF ID1is 0x%d\r\n",flash_id);
        printf("FLASF ID2 is 0x%x\r\n",FLASHID);
        }
        while(1)
        {
        LED0=1;
        delay_ms(500);
        LED0=0;
        LED1=0;
        delay_ms(500);
        LED1=1;
        //printf("FLASF ID is 0x%x\r\n",flash_id);'
        }
       
}

       
       
       
       

正点原子逻辑分析仪DL16劲爆上市
回复

使用道具 举报

530

主题

11万

帖子

34

精华

管理员

Rank: 12Rank: 12Rank: 12

积分
165524
金钱
165524
注册时间
2010-12-1
在线时间
2116 小时
发表于 2020-4-27 01:13:32 | 显示全部楼层
回复

使用道具 举报

6

主题

20

帖子

0

精华

初级会员

Rank: 2

积分
74
金钱
74
注册时间
2019-5-24
在线时间
16 小时
 楼主| 发表于 2020-4-27 14:08:34 | 显示全部楼层
正点原子 发表于 2020-4-27 01:13
你这是SPI 液晶屏么?

就是想利用串口把 SPI读取到的FLASH的ID打印出来
回复

使用道具 举报

530

主题

11万

帖子

34

精华

管理员

Rank: 12Rank: 12Rank: 12

积分
165524
金钱
165524
注册时间
2010-12-1
在线时间
2116 小时
发表于 2020-4-28 01:34:04 | 显示全部楼层
liuhuayi 发表于 2020-4-27 14:08
就是想利用串口把 SPI读取到的FLASH的ID打印出来

是我们开发板么?我们开发板可以直接用我们例程测试下
如果自己做的板子, 示波器看下几个波形对不对先。
比如 CLK, MOSI, MISO等, 还有CS。
我是开源电子网www.openedv.com站长,有关站务问题请与我联系。
正点原子STM32开发板购买店铺http://openedv.taobao.com
正点原子官方微信公众平台,点击这里关注“正点原子”
回复

使用道具 举报

6

主题

20

帖子

0

精华

初级会员

Rank: 2

积分
74
金钱
74
注册时间
2019-5-24
在线时间
16 小时
 楼主| 发表于 2020-4-29 15:31:42 | 显示全部楼层
正点原子 发表于 2020-4-28 01:34
是我们开发板么?我们开发板可以直接用我们例程测试下
如果自己做的板子, 示波器看下几个波形对不对先 ...

是探索者的板子,用例程代码是可以的
回复

使用道具 举报

530

主题

11万

帖子

34

精华

管理员

Rank: 12Rank: 12Rank: 12

积分
165524
金钱
165524
注册时间
2010-12-1
在线时间
2116 小时
发表于 2020-4-30 00:50:24 | 显示全部楼层
liuhuayi 发表于 2020-4-29 15:31
是探索者的板子,用例程代码是可以的

那你自己写的有问题, 对比下吧
可以仿真,对比寄存器配置差异, 最根本的解决办法。
我是开源电子网www.openedv.com站长,有关站务问题请与我联系。
正点原子STM32开发板购买店铺http://openedv.taobao.com
正点原子官方微信公众平台,点击这里关注“正点原子”
回复

使用道具 举报

42

主题

117

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
236
金钱
236
注册时间
2019-8-20
在线时间
90 小时
发表于 2020-4-30 10:25:25 | 显示全部楼层
write_enable(); 表示要置位wel位,
When the program, erase or write status
register instruction has completed, the BUSY bit will be cleared to a 0 state indicating the device is ready
for further instructions.
你没等待直接给了0x90所以有问题
回复

使用道具 举报

6

主题

20

帖子

0

精华

初级会员

Rank: 2

积分
74
金钱
74
注册时间
2019-5-24
在线时间
16 小时
 楼主| 发表于 2020-4-30 18:38:29 | 显示全部楼层
sn3707 发表于 2020-4-30 10:25
write_enable(); 表示要置位wel位,
When the program, erase or write status
register instruction has ...

好的 谢谢 我重新试一试
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则



关闭

原子哥极力推荐上一条 /2 下一条

正点原子公众号

QQ|手机版|OpenEdv-开源电子网 ( 粤ICP备12000418号-1 )

GMT+8, 2025-5-28 01:06

Powered by OpenEdv-开源电子网

© 2001-2030 OpenEdv-开源电子网

快速回复 返回顶部 返回列表