W25Q128是华邦一款容量为128M-bit(16M-byte)的串行NOR Flash储存器。华邦W25Q128 (128M-bit),被组织为65536个可编程的页,每页256bytes。擦除方式分为16页一组(即一个扇区4kbytes),128页一组(即8个扇区32kbytes),256页一组(即16个扇区或1个块64kbytes),或整个芯片擦除。该芯片有4096个可擦除扇区,或256个可擦除块。该芯片支持标准 standard spi,Dual/Quad I/O SPI,高支持的时钟频率为133MHz,在Quad-SPI模式下的连续传输速度可达532MBit/s(66MByte/s)。支持xIP片上执行技术,被广泛用于微控制器的外扩Flash。

        编程即写数据,由于Flash的特性,只能从1编程0,所以写数据之前Flash里面的数据不是0xFF就必须先擦除,然后才能写数据。擦除即将Flash里面的数据恢复为0xFF的过程。

        上电后设备自动处于写禁用状态(Write Enable Latch, WEL为0,WEL是只读位)。在Page Program, Sector Erase, Block Erase, Chip Erase or Write Status Register instruction(页编程、区擦除、块擦除、芯片擦除或者写状态寄存器指令)之前必须先进行写使能指令。在编程、擦除、写状态寄存器指令完成后,WEL自动变成0

W25Q128内部寻址范围

 芯片引脚封装图

后面则需要根据W25Q128手册给的时序图进行编写程序,达到可以进行数据传输的效果。数据手册上有各种各样的指令

Read Data (03h)

Write Status Register-1 (01h), Status Register-2 (31h) & Status Register-3 (11h)

手册里面还有很多,如果需要则自己去详细的读取,下面进行CubeMX的配置

首先配置时钟

在配置端口

我这里采用的是PB12、PB13、PB14、PB15做SPI的四脚控制,然后进行主程序的操修改。

这些都是系统自动生成的,只需要看一下就行了,开始写模块化程序,然后往文件中加入头文件。

w25qxx.c

#include "w25qxx.h"
#include "main.h"
uint16_t W25QXX_TYPE=W25Q128;//默认是W25Q128
//4Kbytes为一个Sector
//16个扇区为1个Block
//W25Q128
//容量为16M字节,共有128个Block,4096个Sector
//SPI2总线读写一个字节
//参数是写入的字节,返回值是读出的字节
uint8_t SPI2_ReadWriteByte(uint8_t TxData)
{uint8_t Rxdata;//定义一个变量RxdataHAL_SPI_TransmitReceive(&hspi2,&TxData,&Rxdata,1,1000);//调用固件库函数收发数据return Rxdata;//返回收到的数据
}
void W25QXX_CS(uint8_t a)//软件控制函数(0为低电平,其他值为高电平)
{if(a==0)HAL_GPIO_WritePin(W25Q128_CS_GPIO_Port, W25Q128_CS_Pin, GPIO_PIN_RESET);else  HAL_GPIO_WritePin(W25Q128_CS_GPIO_Port,  W25Q128_CS_Pin, GPIO_PIN_SET);
}
//初始化SPI FLASH的IO口
uint8_t W25QXX_Init(void)
{uint8_t temp;//定义一个变量tempW25QXX_CS(1);//0片选开启,1片选关闭W25QXX_TYPE = W25QXX_ReadID();//读取FLASH  ID.if(W25QXX_TYPE == W25Q256)//SPI FLASH为W25Q256时才用设置为4字节地址模式{temp = W25QXX_ReadSR(3);//读取状态寄存器3,判断地址模式if((temp&0x01)==0)//如果不是4字节地址模式,则进入4字节地址模式{W25QXX_CS(0);//0片选开启,1片选关闭SPI2_ReadWriteByte(W25X_Enable4ByteAddr);//发送进入4字节地址模式指令W25QXX_CS(1);//0片选开启,1片选关闭}}if(W25QXX_TYPE==W25Q256||W25QXX_TYPE==W25Q128||W25QXX_TYPE==W25Q64||W25QXX_TYPE==W25Q32||W25QXX_TYPE==W25Q16||W25QXX_TYPE==W25Q80)return 0; else return 1;//如果读出ID是现有型号列表中的一个,则识别芯片成功!
}
//读取W25QXX的状态寄存器,W25QXX一共有3个状态寄存器
//状态寄存器1:
//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
//状态寄存器2:
//BIT7  6   5   4   3   2   1   0
//SUS   CMP LB3 LB2 LB1 (R) QE  SRP1
//状态寄存器3:
//BIT7      6    5    4   3   2   1   0
//HOLD/RST  DRV1 DRV0 (R) (R) WPS (R) (R)
//regno:状态寄存器号,范:1~3
//返回值:状态寄存器值
uint8_t W25QXX_ReadSR(uint8_t regno)
{uint8_t byte=0,command=0;switch(regno){case 1:command=W25X_ReadStatusReg1;//读状态寄存器1指令break;case 2:command=W25X_ReadStatusReg2;//读状态寄存器2指令break;case 3:command=W25X_ReadStatusReg3;//读状态寄存器3指令break;default:command=W25X_ReadStatusReg1;//读状态寄存器1指令break;}W25QXX_CS(0);//0片选开启,1片选关闭SPI2_ReadWriteByte(command);//发送读取状态寄存器命令byte=SPI2_ReadWriteByte(0Xff);//读取一个字节W25QXX_CS(1);//0片选开启,1片选关闭return byte;//返回变量byte
}
//写W25QXX状态寄存器
void W25QXX_Write_SR(uint8_t regno,uint8_t  sr)
{uint8_t command=0;switch(regno){case 1:command=W25X_WriteStatusReg1;//写状态寄存器1指令break;case 2:command=W25X_WriteStatusReg2;//写状态寄存器2指令break;case 3:command=W25X_WriteStatusReg3;//写状态寄存器3指令break;default:command=W25X_WriteStatusReg1;break;}W25QXX_CS(0);//0片选开启,1片选关闭SPI2_ReadWriteByte(command);//发送写取状态寄存器命令SPI2_ReadWriteByte(sr);//写入一个字节W25QXX_CS(1);//0片选开启,1片选关闭
}
//W25QXX写使能
//将WEL置位
void W25QXX_Write_Enable(void)
{W25QXX_CS(0);//0片选开启,1片选关闭SPI2_ReadWriteByte(W25X_WriteEnable);//发送写使能W25QXX_CS(1);//0片选开启,1片选关闭
}
//W25QXX写禁止
//将WEL清零
void W25QXX_Write_Disable(void)
{W25QXX_CS(0);//0片选开启,1片选关闭SPI2_ReadWriteByte(W25X_WriteDisable);//发送写禁止指令W25QXX_CS(1);//0片选开启,1片选关闭
}
//读取芯片ID
//高8位是厂商代号(本程序不判断厂商代号)
//低8位是容量大小
//0XEF13型号为W25Q80
//0XEF14型号为W25Q16
//0XEF15型号为W25Q32
//0XEF16型号为W25Q64
//0XEF17型号为W25Q128(目前洋桃2号开发板使用128容量芯片)
//0XEF18型号为W25Q256
uint16_t W25QXX_ReadID(void)
{uint16_t Temp = 0;W25QXX_CS(0);//0片选开启,1片选关闭SPI2_ReadWriteByte(0x90);//发送读取ID命令SPI2_ReadWriteByte(0x00);SPI2_ReadWriteByte(0x00);SPI2_ReadWriteByte(0x00);Temp|=SPI2_ReadWriteByte(0xFF)<<8;Temp|=SPI2_ReadWriteByte(0xFF);W25QXX_CS(1);//0片选开启,1片选关闭return Temp;
}
//读取SPI FLASH
//在指定地址开始读取指定长度的数据
//pBuffer:数据存储区
//ReadAddr:开始读取的地址(24bit)
//NumByteToRead:要读取的字节数(最大65535)
void W25QXX_Read(uint8_t* pBuffer,uint32_t  ReadAddr,uint16_t NumByteToRead)
{uint16_t i;W25QXX_CS(0);//0片选开启,1片选关闭SPI2_ReadWriteByte(W25X_ReadData);//发送读取命令if(W25QXX_TYPE==W25Q256)//如果是W25Q256的话地址为4字节的,要发送最高8位{SPI2_ReadWriteByte((uint8_t)((ReadAddr)>>24));}SPI2_ReadWriteByte((uint8_t)((ReadAddr)>>16));//发送24bit地址SPI2_ReadWriteByte((uint8_t)((ReadAddr)>>8));SPI2_ReadWriteByte((uint8_t)ReadAddr);for(i=0;i<NumByteToRead;i++){pBuffer[i]=SPI2_ReadWriteByte(0XFF);//循环读数}W25QXX_CS(1);//0片选开启,1片选关闭
}
//SPI在一页(0~65535)内写入少于256个字节的数据
//在指定地址开始写入最大256字节的数据
//pBuffer:数据存储区
//WriteAddr:开始写入的地址(24bit)
//NumByteToWrite:要写入的字节数(最大256),该数不应该超过该页的剩余字节数!!!
void W25QXX_Write_Page(uint8_t*  pBuffer,uint32_t WriteAddr,uint16_t  NumByteToWrite)
{uint16_t i;W25QXX_Write_Enable();//SET WELW25QXX_CS(0);//0片选开启,1片选关闭SPI2_ReadWriteByte(W25X_PageProgram);//发送写页命令if(W25QXX_TYPE==W25Q256)//如果是W25Q256的话地址为4字节的,要发送最高8位{SPI2_ReadWriteByte((uint8_t)((WriteAddr)>>24));}SPI2_ReadWriteByte((uint8_t)((WriteAddr)>>16));//发送24bit地址SPI2_ReadWriteByte((uint8_t)((WriteAddr)>>8));SPI2_ReadWriteByte((uint8_t)WriteAddr);for(i=0;i<NumByteToWrite;i++)SPI2_ReadWriteByte(pBuffer[i]);//循环写数W25QXX_CS(1);//0片选开启,1片选关闭W25QXX_Wait_Busy();//等待写入结束
}
//无检验写SPI FLASH
//必须确保所写的地址范围内的数据全部为0XFF,否则在非0XFF处写入的数据将失败!
//具有自动换页功能
//在指定地址开始写入指定长度的数据,但是要确保地址不越界!
//pBuffer:数据存储区
//WriteAddr:开始写入的地址(24bit)
//NumByteToWrite:要写入的字节数(最大65535)
//CHECK OK
void W25QXX_Write_NoCheck(uint8_t*  pBuffer,uint32_t WriteAddr,uint16_t  NumByteToWrite)
{uint16_t pageremain;pageremain=256-WriteAddr%256; //单页剩余的字节数if(NumByteToWrite<=pageremain)pageremain=NumByteToWrite;//不大于256个字节while(1){W25QXX_Write_Page(pBuffer,WriteAddr,pageremain);if(NumByteToWrite==pageremain)break;//写入结束了else //NumByteToWrite>pageremain{pBuffer+=pageremain;WriteAddr+=pageremain;NumByteToWrite-=pageremain;            //减去已经写入了的字节数if(NumByteToWrite>256)pageremain=256; //一次可以写入256个字节else pageremain=NumByteToWrite;     //不够256个字节了}};
}
//写SPI FLASH
//在指定地址开始写入指定长度的数据
//该函数带擦除操作!
//pBuffer:数据存储区
//WriteAddr:开始写入的地址(24bit)
//NumByteToWrite:要写入的字节数(最大65535)
uint8_t W25QXX_BUFFER[4096];
void W25QXX_Write(uint8_t* pBuffer,uint32_t  WriteAddr,uint16_t NumByteToWrite)
{uint32_t secpos;uint16_t secoff;uint16_t secremain;uint16_t i;uint8_t* W25QXX_BUF;W25QXX_BUF=W25QXX_BUFFER;secpos=WriteAddr/4096;//扇区地址secoff=WriteAddr%4096;//在扇区内的偏移secremain=4096-secoff;//扇区剩余空间大小//printf("ad:%X,nb:%X\r\n",WriteAddr,NumByteToWrite);//测试用if(NumByteToWrite<=secremain)secremain=NumByteToWrite;//不大于4096个字节while(1){W25QXX_Read(W25QXX_BUF,secpos*4096,4096);//读出整个扇区的内容for(i=0;i<secremain;i++)//校验数据{if(W25QXX_BUF[secoff+i]!=0XFF)break;//需要擦除}if(i<secremain)//需要擦除{W25QXX_Erase_Sector(secpos);//擦除这个扇区for(i=0;i<secremain;i++)//复制{W25QXX_BUF[i+secoff]=pBuffer[i];}W25QXX_Write_NoCheck(W25QXX_BUF,secpos*4096,4096);//写入整个扇区}else  W25QXX_Write_NoCheck(pBuffer,WriteAddr,secremain);//写已经擦除了的,直接写入扇区剩余区间.if(NumByteToWrite==secremain)break;//写入结束了else//写入未结束{secpos++;//扇区地址增1secoff=0;//偏移位置为0pBuffer+=secremain;  //指针偏移WriteAddr+=secremain;//写地址偏移NumByteToWrite-=secremain;//字节数递减if(NumByteToWrite>4096)secremain=4096;//下一个扇区还是写不完else  secremain=NumByteToWrite;//下一个扇区可以写完了}};
}
//擦除整个芯片
//等待时间超长...
void W25QXX_Erase_Chip(void)
{W25QXX_Write_Enable();//SET WELW25QXX_Wait_Busy();//等待忙状态W25QXX_CS(0);//0片选开启,1片选关闭SPI2_ReadWriteByte(W25X_ChipErase);//发送片擦除命令W25QXX_CS(1);//0片选开启,1片选关闭W25QXX_Wait_Busy();//等待芯片擦除结束
}
//擦除一个扇区
//Dst_Addr:扇区地址 根据实际容量设置
//擦除一个扇区的最少时间:150ms
void W25QXX_Erase_Sector(uint32_t Dst_Addr)
{Dst_Addr*=4096;W25QXX_Write_Enable();//SET WELW25QXX_Wait_Busy();W25QXX_CS(0);//0片选开启,1片选关闭SPI2_ReadWriteByte(W25X_SectorErase);//发送扇区擦除指令if(W25QXX_TYPE==W25Q256)//如果是W25Q256的话地址为4字节的,要发送最高8位{SPI2_ReadWriteByte((uint8_t)((Dst_Addr)>>24));}SPI2_ReadWriteByte((uint8_t)((Dst_Addr)>>16));//发送24bit地址SPI2_ReadWriteByte((uint8_t)((Dst_Addr)>>8));SPI2_ReadWriteByte((uint8_t)Dst_Addr);W25QXX_CS(1);//0片选开启,1片选关闭W25QXX_Wait_Busy();//等待擦除完成
}
//等待空闲
void W25QXX_Wait_Busy(void)
{while((W25QXX_ReadSR(1)&0x01)==0x01);//等待BUSY位清空
}
//进入掉电模式
void W25QXX_PowerDown(void)
{W25QXX_CS(0);//0片选开启,1片选关闭SPI2_ReadWriteByte(W25X_PowerDown);//发送掉电命令 0xB9W25QXX_CS(1);//0片选开启,1片选关闭delay_us(3);//等待TPD
}
//唤醒
void W25QXX_WAKEUP(void)
{W25QXX_CS(0);//0片选开启,1片选关闭SPI2_ReadWriteByte(W25X_ReleasePowerDown);//发送电源唤醒指令 0xABW25QXX_CS(1);//0片选开启,1片选关闭delay_us(3);//等待TRES1
}

w25q128.h

#ifndef W25Q128_W25QXX_H_
#define W25Q128_W25QXX_H_#include "stm32f1xx_hal.h" //HAL库文件声明
#include "../delay/delay.h"//25系列FLASH芯片厂商与容量代号(厂商代号EF)
#define W25Q80    0XEF13
#define W25Q16    0XEF14
#define W25Q32    0XEF15
#define W25Q64    0XEF16
#define W25Q128   0XEF17
#define W25Q256 0XEF18
#define EX_FLASH_ADD 0x000000 //W25Q128的地址是24位宽
extern uint16_t W25QXX_TYPE;//定义W25QXX芯片型号
extern SPI_HandleTypeDef hspi2;
//
//指令表
#define W25X_WriteEnable             0x06
#define W25X_WriteDisable            0x04
#define W25X_ReadStatusReg1      0x05
#define W25X_ReadStatusReg2      0x35
#define W25X_ReadStatusReg3      0x15
#define W25X_WriteStatusReg1         0x01
#define W25X_WriteStatusReg2         0x31
#define W25X_WriteStatusReg3     0x11
#define W25X_ReadData             0x03
#define W25X_FastReadData         0x0B
#define W25X_FastReadDual         0x3B
#define W25X_PageProgram          0x02
#define W25X_BlockErase              0xD8
#define W25X_SectorErase          0x20
#define W25X_ChipErase            0xC7
#define W25X_PowerDown            0xB9
#define W25X_ReleasePowerDown    0xAB
#define W25X_DeviceID             0xAB
#define W25X_ManufactDeviceID    0x90
#define W25X_JedecDeviceID           0x9F
#define W25X_Enable4ByteAddr         0xB7
#define W25X_Exit4ByteAddr        0xE9
uint8_t SPI2_ReadWriteByte(uint8_t  TxData);//SPI2总线底层读写
void W25QXX_CS(uint8_t a);//W25QXX片选引脚控制
uint8_t W25QXX_Init(void);//初始化W25QXX函数
uint16_t  W25QXX_ReadID(void);//读取FLASH ID
uint8_t W25QXX_ReadSR(uint8_t regno);//读取状态寄存器
void W25QXX_4ByteAddr_Enable(void);//使能4字节地址模式
void W25QXX_Write_SR(uint8_t regno,uint8_t  sr);//写状态寄存器
void W25QXX_Write_Enable(void);//写使能
void W25QXX_Write_Disable(void);//写保护
void W25QXX_Write_NoCheck(uint8_t*  pBuffer,uint32_t WriteAddr,uint16_t  NumByteToWrite);//无检验写SPI FLASH
void W25QXX_Read(uint8_t* pBuffer,uint32_t  ReadAddr,uint16_t NumByteToRead);//读取flash
void W25QXX_Write(uint8_t* pBuffer,uint32_t  WriteAddr,uint16_t NumByteToWrite);//写入flash
void W25QXX_Erase_Chip(void);//整片擦除
void W25QXX_Erase_Sector(uint32_t  Dst_Addr);//扇区擦除
void W25QXX_Wait_Busy(void);//等待空闲
void W25QXX_PowerDown(void);//进入掉电模式
void W25QXX_WAKEUP(void);//唤醒#endif /* W25Q128_W25QXX_H_ */

这时你会发现,程序很多,但只需要理解那些是需要用的那些其重点函数。

这里面用到了delay的程序

delay.c

#include "delay.h"void delay_us(uint32_t us) //利用CPU循环实现的非精准应用的微秒延时函数
{uint32_t delay = (HAL_RCC_GetHCLKFreq() / 8000000 * us); //使用HAL_RCC_GetHCLKFreq()函数获取主频值,经算法得到1微秒的循环次数while (delay--); //循环delay次,达到1微秒延时
}

delay.h

#ifndef DELAY_DELAY_H_
#define DELAY_DELAY_H_#include "stm32f1xx_hal.h" //HAL库文件声明
void delay_us(uint32_t us); //C文件中的函数声明#endif /* DELAY_DELAY_H_ */

而我们在使用这些的时候,则是通过应用程序方面去进行编写的。

在main.c下面创一个自定义的uint8_t的自变量数组,数组是需要根据自己的需求来进行使用的。

就这样可以进行操作了。

总结:

如果想深入理解就去看手册,掌握使用就会用就行。

STM32CubeMX之SPI闪存芯片W25Q128相关推荐

  1. STM32CUBEMX(13)--SPI,W25Q128外部Flash移植

    STM32CUBEMX--SPI,W25Q128外部Flash移植 概述 视频教学 完整代码下载 硬件准备 选择芯片型号 配置时钟源 配置时钟树 串口配置 SPI配置 接线方式 生成工程设置 生成代码 ...

  2. SPI应用——W25Q128串行FLASH

    一.FLASH存储器介绍 FLASH存储器又称闪存,它与EEPROM都是掉电后数据不丢失的存储器,但FLASH存储器容量普遍大于EEPROM,现在基本取代了它的地位.在存储控制上,最主要的区别是FLA ...

  3. SPI与W25Q128

    SPI与W25Q128 1 SPI简介 2 SPI协议 3 STM32F4xx SPI控制器 4 STM32固件库SPI函数接口说明 4.1 配置SPI的GPIO引脚 (SPI - W25Q128) ...

  4. 基于ART-PI SPI驱动W25Q128

    吐槽和避坑 弄spi弄了一天,使用的开发板是art-pi家的h750,本来准备直接用rt-thread studio生成的,但是不知道怎么回事配置出的工程文件都有问题,不是线程运行不了就是缺少定义,所 ...

  5. 【STM32CubeMx你不知道的那些事】第十章:STM32CubeMx的SPI外置FLASH(W25Q128)+文件系统(FATFS)+虚拟U盘

      这一张我们主要讲解一下STM32CUBEMX新版本 片外FLASH(W25Q128)+FATFS文件系统+虚拟U盘. 一.准备工作 这里我们要想配置SPI和文件系统 并验证需要的准备工作如下: 1 ...

  6. STM32F103配合STM32CubeMX实现SPI读写flash

    本人采用的是正点原子的精英STM32F103开发板,其包含一块W25Q128型号的flash芯片.该flash与STM32F103的SPI2相连. 下面根据正点原子提供的开发指南文档,实现FreeRT ...

  7. STC8通过SPI读写W25Q128

    这几天在看SPI总线,应用主要是围绕W25Q128的读写进行的. 网上关于W25Q的程序可以说非常多,大多是以STM32为主控芯片进行的.无妨.这块FLASH的寄存器是基本固定的,但是只有英文手册确实 ...

  8. STM32之SPI和W25Q128

    目录 SPI 介绍 SPI 物理架构 SPI 工作原理 SPI 工作模式 W25Q128 介绍 W25Q128 存储架构 W25Q128 常用指令 W25Q128 状态寄存器 W25Q128 常见操作 ...

  9. stm32 cubemx usb spi flash w25q128 u盘调试笔记

    真的太简单了,十分钟就搞定 参考文章 我卡住了几天,最后发现delay函数的问题,去掉就好了.(评论大佬解释了这一现象) 步骤如下 使用cube mx 生成基本代码 调试spi flash 调试usb ...

最新文章

  1. java交通工具的类继承代码_Java作业-交通工具继承
  2. 关于[入行几年后,你的未来应该在哪里]的思考
  3. ITK:计算曲率各向异性扩散
  4. Roadblocks(次短路经)
  5. js(Dom+Bom)第三天(2)
  6. 江苏省计算机等级知识,江苏省计算机二级考试基础知识_计算机基础练习题
  7. jQuery的name选择器 模糊匹配
  8. Jenkins插件Gerrit Trigger配置,实现change-merged时自动触发Jenkins工程build
  9. 含指数函数的不定积分方法归纳
  10. 立创商城PCB库使用说明
  11. Qt操作Word文档
  12. oracle插入报错-Caused by: java.sql.SQLException: 无效的列类型: 16
  13. 为什么静下心来阅读会是一种奢侈?
  14. 微信小程序转盘demo
  15. 如何查看Outlook搜索出的邮件所在的文件夹
  16. iOS 后台运行 Background Task与Background Mode
  17. C语言调用libusb访问USB驱动
  18. 佳能相机照片误删怎么恢复?看看我是如何在10分钟内解决的
  19. android版本过高导致下载软件后无法自动安装
  20. 牟长青:浅谈如何提高网站PR值

热门文章

  1. 2021年考思科认证CCIE还有必要吗
  2. 开源报表工具python_开源web报表工具哪家强?5款最优软件
  3. YGGSEA 购买了价值 25 万美金的 Nitro League 游戏资产
  4. .NET 云原生架构师训练营(模块二 基础巩固 RabbitMQ HelloWorld)--学习笔记
  5. 贝叶斯--旧金山犯罪分类预测和电影评价好坏 demo
  6. 基于冲突搜索的多机器人路径规划(Matlab代码实现)
  7. C#编程UDP通信过程中出现【远程主机强迫关闭了一个现有的连接0x80004005】的解决方法
  8. ABAP中的向上取整函数Ceil( )使用注意事项(decfloat16 decfloat34)
  9. 2023五一出游数据报告.pdf(附下载链接)
  10. C++的字符串拷贝函数