STM32 SPI2读W25Q128驱动

最新更新时间:2019-03-07来源: eefocus关键字:STM32  SPI2  读W25Q128驱动 手机看文章 扫描二维码
随时随地手机看文章

//SPI1 读写一个字节

//TxData:要写入的字节

//返回值:读取到的字节

u8 SPI2_ReadWriteByte(u8 TxData)

{  

  while (SPI_I2S_GetFlagStatus(SPI2, SPI_I2S_FLAG_TXE) == RESET){}//等待发送区空  

SPI_I2S_SendData(SPI2, TxData); //通过外设SPIx发送一个byte  数据

  while (SPI_I2S_GetFlagStatus(SPI2, SPI_I2S_FLAG_RXNE) == RESET){} //等待接收完一个byte  

return SPI_I2S_ReceiveData(SPI2); //返回通过SPIx最近接收的数据 

     

}



/*uint8_t sFLASH_SendByte(uint8_t byte)

{

  //!< Loop while DR register in not emplty 

  while (SPI_I2S_GetFlagStatus(sFLASH_SPI, SPI_I2S_FLAG_TXE) == RESET);



  //!< Send byte through the SPI1 peripheral 

  SPI_I2S_SendData(sFLASH_SPI, byte);



  //!< Wait to receive a byte 

  while (SPI_I2S_GetFlagStatus(sFLASH_SPI, SPI_I2S_FLAG_RXNE) == RESET);



  //!< Return the byte read from the SPI bus 

  return SPI_I2S_ReceiveData(sFLASH_SPI);

}

*/



//以下是SPI模块的初始化代码,配置成主机模式  

//SPI口初始化

//这里针是对SPI1的初始化

void SPI2_Init(void)

{  

  GPIO_InitTypeDef  GPIO_InitStructure;

  SPI_InitTypeDef  SPI_InitStructure;


  RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE);//使能GPIOB时钟

  RCC_APB1PeriphClockCmd(RCC_APB1Periph_SPI2, ENABLE);//使能SPI2时钟

 

  //GPIOFB3,4,5初始化设置

  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_13|GPIO_Pin_14|GPIO_Pin_15;//PB3~5复用功能输出 

  GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;

  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;

  GPIO_Init(GPIOB, &GPIO_InitStructure);//初始化


//这里只针对SPI口初始化

RCC_APB1PeriphResetCmd(RCC_APB1Periph_SPI2,ENABLE);//复位SPI1

RCC_APB1PeriphResetCmd(RCC_APB1Periph_SPI2,DISABLE);//停止复位SPI1



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_2; //定义波特率预分频的值:波特率预分频值为256

SPI_InitStructure.SPI_FirstBit = SPI_FirstBit_MSB; //指定数据传输从MSB位还是LSB位开始:数据传输从MSB位开始

SPI_InitStructure.SPI_CRCPolynomial = 7; //CRC值计算的多项式

SPI_Init(SPI2, &SPI_InitStructure);  //根据SPI_InitStruct中指定的参数初始化外设SPIx寄存器

 

SPI_Cmd(SPI2, ENABLE); //使能SPI外设



SPI2_ReadWriteByte(0xff);//启动传输  

}  





void W25QXX_Init(void)

  GPIO_InitTypeDef  GPIO_InitStructure;

 

  RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE);//使能GPIOB时钟

 //GPIOB14



  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_12;//PB12

  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;//输出

  GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;//100MHz

  GPIO_Init(GPIOB, &GPIO_InitStructure);//初始化

    

//W25QXX_CS=1; //SPI FLASH不选中

GPIO_SetBits(GPIOB,GPIO_Pin_12);

SPI2_Init();   //初始化SPI



//W25QXX_TYPE=W25QXX_ReadID(); //读取FLASH ID.

}  



#define EN_FLASH() GPIO_ResetBits(GPIOB,GPIO_Pin_12)

#define DIS_FLASH() GPIO_SetBits(GPIOB,GPIO_Pin_12);

//读取W25QXX的状态寄存器

//BIT7  6   5   4   3   2   1   0

//SPR   RV  TB BP2 BP1 BP0 WEL BUSY

//SPR:默认0,状态寄存器保护位,配合WP使用

//TB,BP2,BP1,BP0:FLASH区域写保护设置

//WEL:写使能锁定

//BUSY:忙标记位(1,忙;0,空闲)

//默认:0x00



u8 W25QXX_ReadSR(void)   

{  

u8 byte=0;   

//W25QXX_CS=0;                            //使能器件  

  //GPIO_ResetBits(GPIOD,GPIO_Pin_0);

  EN_FLASH(); 

SPI2_ReadWriteByte(W25X_ReadStatusReg);    //发送读取状态寄存器命令    

byte=SPI2_ReadWriteByte(0Xff);             //读取一个字节  

//W25QXX_CS=1;                            //取消片选     

DIS_FLASH();

return byte;   



//写W25QXX状态寄存器

//只有SPR,TB,BP2,BP1,BP0(bit 7,5,4,3,2)可以写!!!

void W25QXX_Write_SR(u8 sr)   

{   

//W25QXX_CS=0;                            //使能器件   

EN_FLASH();

SPI2_ReadWriteByte(W25X_WriteStatusReg);   //发送写取状态寄存器命令    

SPI2_ReadWriteByte(sr);               //写入一个字节  

//W25QXX_CS=1;                            //取消片选          

  DIS_FLASH(); 

}  



//W25QXX写使能 

//将WEL置位   

void W25QXX_Write_Enable(void)   

{

//W25QXX_CS=0;                            //使能器件   

  EN_FLASH();  

  SPI2_ReadWriteByte(W25X_WriteEnable);      //发送写使能  

//W25QXX_CS=1;                            //取消片选          

  DIS_FLASH();



//等待空闲

void W25QXX_Wait_Busy(void)   

{   

while((W25QXX_ReadSR()&0x01)==0x01)   IWDG_ReloadCounter();   //feed dog ;   // 等待BUSY位清空



//擦除一个扇区

//Dst_Addr:扇区地址 根据实际容量设置

//擦除一个山区的最少时间:150ms

void W25QXX_Erase_Sector(u32 Dst_Addr)   

{  

//监视falsh擦除情况,测试用   

    Dst_Addr*=4096;

    W25QXX_Write_Enable();                  //SET WEL  

    W25QXX_Wait_Busy();   

  //W25QXX_CS=0;                            //使能器件   

    EN_FLASH();

 SPI2_ReadWriteByte(W25X_SectorErase);      //发送扇区擦除指令 

    SPI2_ReadWriteByte((u8)((Dst_Addr)>>16));  //发送24bit地址    

    SPI2_ReadWriteByte((u8)((Dst_Addr)>>8));   

    SPI2_ReadWriteByte((u8)Dst_Addr);  

 DIS_FLASH();                          //取消片选          

    W25QXX_Wait_Busy();     //等待擦除完成

}  

 



//SPI在一页(0~65535)内写入少于256个字节的数据

//在指定地址开始写入最大256字节的数据

//pBuffer:数据存储区

//WriteAddr:开始写入的地址(24bit)

//NumByteToWrite:要写入的字节数(最大256),该数不应该超过该页的剩余字节数!!!  

void W25QXX_Write_Page(u8* pBuffer,u32 WriteAddr,u16 NumByteToWrite)

{

  u16 i;  

    W25QXX_Write_Enable();                  //SET WEL 

//W25QXX_CS=0;                            //使能器件   

    EN_FLASH();

 SPI2_ReadWriteByte(W25X_PageProgram);      //发送写页命令   

    SPI2_ReadWriteByte((u8)((WriteAddr)>>16)); //发送24bit地址    

    SPI2_ReadWriteByte((u8)((WriteAddr)>>8));   

    SPI2_ReadWriteByte((u8)WriteAddr);   

    for(i=0;i

//W25QXX_CS=1;                            //取消片选 

DIS_FLASH();

W25QXX_Wait_Busy();   //等待写入结束

}



//读取SPI FLASH  

//在指定地址开始读取指定长度的数据

//pBuffer:数据存储区

//ReadAddr:开始读取的地址(24bit)

//NumByteToRead:要读取的字节数(最大65535)

void W25QXX_Read(u8* pBuffer,u32 ReadAddr,u16 NumByteToRead)   

  u16 i;      

EN_FLASH();                            //使能器件   

    SPI2_ReadWriteByte(W25X_ReadData);         //发送读取命令   

    SPI2_ReadWriteByte((u8)((ReadAddr)>>16));  //发送24bit地址    

    SPI2_ReadWriteByte((u8)((ReadAddr)>>8));   

    SPI2_ReadWriteByte((u8)ReadAddr);   

    for(i=0;i

        pBuffer[i]=SPI2_ReadWriteByte(0XFF);   //循环读数  

    }

DIS_FLASH();          

}  


关键字:STM32  SPI2  读W25Q128驱动 编辑:什么鱼 引用地址:http://news.eeworld.com.cn/mcu/2019/ic-news030743348.html

上一篇:STM32通信接口(一)串口
下一篇:基于STM32的DS18B20驱动

推荐阅读

STM32 SPI硬件模式
反复试验,发现SPI_NSS引脚的自动硬件控制与想象的不同,无论是否外加上拉,只要一使能SPI,SPI_Cmd(SPI1, ENABLE); SPI_NSS引脚就一直处于低电平,直到SPI_Cmd(SPI1, DISABLE);这个需要用程序来控制。  而用过其他芯片则是发送完成自动会拉高,这点是要注意的我说的就是做主机的时候SPI_SSOutputCmd(SPIx,ENABLE) 在soft模式时这句话有必要吗?我的理解是当hard模式,需要multimaster的时候,才应该要开启这个output功能,这点从我的截图上可以看出。我觉得,这里只要把SPI_InitStructure
发表于 2022-02-21
<font color='red'>STM32</font> SPI硬件模式
STM32 timer input filter
STM32的定时器输入通道都有一个滤波单元,分别位于每个输入通路上(下图中的黄色框)和外部触发输入通路上(下图中的兰色框),它们的作用是滤除输入信号上的高频干扰。具体操作原理如下:在TIMx_CR1中的CKD[1:0]可以由用户设置对输入信号的采样频率基准,有三种选择:1)采样频率基准fDTS=定时器输入频率fCK_INT2)采样频率基准fDTS=定时器输入频率fCK_INT/23)采样频率基准fDTS=定时器输入频率fCK_INT/4然后使用上述频率作为基准对输入信号进行采样,当连续采样到N次个有效电平时,认为一次有效的输入电平。实际的采样频率和采样次数可以由用户程序根据需要选择;外部触发输入通道的滤波参数在从模式控制寄存器(TI
发表于 2022-02-21
<font color='red'>STM32</font> timer input filter
关于STM32影子寄存器和预装载寄存器和TIM_ARRPreloadConfig
本文的说明依据STM32参考手册(RM0008)第10版:英文:http://www.st.com/stonline/products/literature/rm/13902.pdf中译文:http://www.stmicroelectronics.com.cn/stonline/mcu/images/STM32_RM_CH_V10_1.pdf在STM32参考手册的第13、14章中,都有一张定时器的框图,下面是第14章中定时器框图的局部,图中黄色框所示的是auto-reload register,在下面的第14.3.2节"Counter Modes"就解释了auto-reload register的用法。在图中可
发表于 2022-02-21
关于<font color='red'>STM32</font>影子寄存器和预装载寄存器和TIM_ARRPreloadConfig
stm32 Fdts
发表于 2022-02-21
<font color='red'>stm32</font> Fdts
STM32定时器输出比较模式
OCx与OCxREF和CCxP之间的关系初学STM32,我这个地方卡了很久,现在终于有些明白了,现在把我的理解写下与大家共享,如果有不对的地方,还请指出。-----------------------------------------------------------------------------------------------------------------------TIM_OCMode选择定时器模式。该参数取值见下表:TIM_OCInitStructure.TIM_Pulse = CCR1_Val; //设置跳变值,当计数器计数到这个值时,电平发生跳变TIM_OC2PreloadConfig(TIM3, TI
发表于 2022-02-18
<font color='red'>STM32</font>定时器输出比较模式
stm32库函数学习篇
两天学习了一下stm32通用定时器的输入捕获功能。在网上看到很多网友说触发中断程序进不了,于是自己也测试了个小程序,还好能够进入中断。呵呵~ 实现功能:PA8随意延时驱动led灯闪烁,并且将PA8用杜邦线连接到PA7口,PA7是通用定时器TIM3的2通道,在TIM3_CH2触发中断程序中取反连接到PD2口的led灯,指示中断程序运行,并且每次进入中断后改变触发捕获的极性。实现两个led灯会交替闪烁。 先有必要了解stm32定时器的输入触发模块,如下图:需要注意的是,一眼望去一个定时器似乎有8个通道,左边四个,右边四个,但其实左边和右边是共用相同的IO引脚,所以名称标注是一模一样。也就是说,每个通用定时器都只有四个
发表于 2022-02-18
<font color='red'>stm32</font>库函数学习篇
小广播
实战 培训 开发板 精华推荐

何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件
电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2022 EEWORLD.com.cn, Inc. All rights reserved