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驱动 引用地址:STM32 SPI2读W25Q128驱动

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

推荐阅读最新更新时间:2024-03-16 16:24

关于STM32中GPIO的8种工作模式
1 综述 I/O口是单片机中非常常用的外设,STM32的I/O口有8种状态,虽然一直在使用过程中没有遇到什么问题,但是一直都不是很清楚,因此这里做一个总结(实际上这里的概念也是和STM8等其他单片机,理解了这8中状态,也就基本上理解了大部分I/O口)。 2 庐山真面目 我们在库文件中的 stm32f10x_gpio.h 中可以看到如下代码: typedefenum { GPIO_Mode_AIN = 0x0, GPIO_Mode_IN_FLOATING = 0x04, GPIO_Mode_IPD = 0x28, GPIO_Mode_IPU = 0x48, GPIO_Mode_Out_OD = 0x14
[单片机]
关于<font color='red'>STM32</font>中GPIO的8种工作模式
STM32通用定时器的输入捕获(实例:输入捕获)
通用定时器输入捕获概述 输入捕获的工作原理 在通用定时器框图中,主要涉及到最顶上的一部分(计数时钟的选择)、中间部分(时基单元)、左下部分(输入捕获)这三个部分。这里主要讲解一下左下部分(输入捕获),其他两个部分可以参考文章:【STM32】通用定时器的基本原理(实例:定时器中断)。 输入捕获模式可以用来测量脉冲宽度或者测量频率。STM32的定时器,除了TIM6、TIM7,其他的定时器都有输入捕获的功能。下面以一个简单的脉冲输入为例,简单地讲述一下输入捕获用于测量脉冲宽度的工作原理: 先设置输入捕获为上升沿检测,记录发生上升沿时TIMx_CNT的值。然后配置捕获信号为下降沿捕获,当下降沿到来的时候
[单片机]
<font color='red'>STM32</font>通用定时器的输入捕获(实例:输入捕获)
基于STM32战舰开发板的USMART调试组件的使用
USMART调试组件是什么? USMART是正点原子团队为其STM32开发平台开发的一种类似linux的shell的调试工具。具体工作过程是通过串口发送命令给单片机,然后单片机收到命令之后调用单片机里面对应的相关函数,并执行,同时支持返回结果。 USMART是干什么的? USMART的功能就是改变了函数参数修改的方式,我们以前总是“修改函数参数- 下载到开发板中执行”,但是使用USMART之后,我们不用在那样做了,而是“用串口进行函数参数的修改- 开发板执行相应操作”。 在我看来,USMART就是一个函数调试助手,可以从串口发送参数也可以从串口接收函数返回值。 USMART的作用机理 这里我们要明白一下,函数的存储方式
[单片机]
基于<font color='red'>STM32</font>战舰开发板的USMART调试组件的使用
STM32 printf函数的调用
stm32中调用 printf()函数,实质上添加对应的 int fputc(int ch, FILE *f) 函数 (如 fputc对应目标是串口则printf输在串口上,如 fputc对应目标是LCD则是输在LCD) 具体步骤如下: 1、 首先要在main文件中包含 stdio.h (标准输入输出头文件) 2、在main文件中重定义函数 //加入以下代码,支持printf函数,而不需要选择use MicroLIB #if 1 #pragma import(__use_no_semihosting) //标准库需要的支持函数 struct __FILE { int handle; }; FILE __stdou
[单片机]
STM32学习笔记1 IO口学习
STM32的IO口可以由软件配置成8种模式: 1,输入浮空 2,输入上拉 3,输入下拉 4,模拟输入 5,开漏输出 6,推挽输出 7,推挽复用功能 8,开漏复用功能 每个IO口可以自由编程,单IO口寄存器必须要按32位字被访问。 这里就是寄存器不能位操作咯 STM32的每个IO端口都有7个寄存器来控制。他们分别是:配置模式的2个32位的端口配置寄存器CRL和CRH;2个32位的数据寄存器IDR和ODR;1个32位的置位/复位寄存器BSRR;一个16位的复位寄存器BRR;1个32位的锁存寄存器LCKR;这里我们仅介绍常用 的几个寄存器,我们常用的IO端口寄存器只有4个:CRL、CRH、IDR、ODR。 STM32的CRL控制
[单片机]
STM32硬件乘法器——另与MSP430硬件乘法器区别
用过MSP430的人都知道,它是带有硬件乘法器的,可以设计进行简单的FFT计算,但是它是单独的模块,需要写入两个乘数,然后再到结果中取结果。而我们大多数时候根本就没必要去这么做,就当做51一样,软解乘法,很耗费时间。这2个MCU没有把乘法器内嵌到MCU内核MUL指令中。 而STM32包括各种使用Coretex M内核的处理器时真实的把硬件乘法器关联到MUL中了,所以你就和平时一样写乘法程序计算,实际上就是在调用乘法器。令人高兴的是,STM32能够进行单周期的乘除法,所以比软解乘除法要快几十倍呢。
[单片机]
STM32——串口通信升级版(队列方式)
#ifndef _USART_QUEUE_H_ #define _USART_QUEUE_H_ #include type.h typedef enum { USART_QUEUE_EMPTY = 0, USART_QUEUE_FULL = 1, USART_QUEUE_OK = 2, } usart_queue_status_t; #define USART_QUEUE_SIZE 1024 typedef struct { uint16_t front; uint16_t rear; uint16_t size; char data ; } usart_queue_t; extern usa
[单片机]
网络通信之UIP在STM32上的移植
1、什么是UIP协议栈: UIP是一种适用于小型嵌入式通信的TCP/IP协议栈,由瑞典计算机科学院的人开发编写的,它去掉了完整的TCP/IP中不常用的功能,简化了通讯过程,但其保留了网络通信中必须使用的协议,把设计重点放在IP、TCP和ARP协议的实现上。简单易用、占用资源少是它的设计特点。 2、协议栈接口 UIP协议栈通过一系列接口函数与底层函数和上层应用程序通信,在两者之中,UIP接口函数处于中间。 其中UIP提供两个个接口函数给底层系统。分别是 (1)uip_input() 当网卡驱动收到一个输入包时,将放入全局缓冲区uip_buf中,包的大小由全局变量uip_len约束。同时将调用 uip_input() 函数,这个函
[单片机]
网络通信之UIP在<font color='red'>STM32</font>上的移植
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

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

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