【linux设备驱动程序|W25Qxx nor flash驱动学习】如下所示是原子哥提供的w25qxx的驱动,作为学习参考非常实用,驱动比较通用,注释清晰,方便移植:
#include "w25qxx.h"
#include "spi.h"
#include "delay.h"
#include "usart.h"
#include "stm32f4xx_hal_gpio.h"u16 W25QXX_TYPE=W25Q256;
//默认是W25Q256//4Kbytes为一个Sector
//16个扇区为1个Block
//W25Q256
//容量为32M字节,共有512个Block,8192个Sector //初始化SPI FLASH的IO口
void W25QXX_Init(void)
{
u8 temp;
GPIO_InitTypeDef GPIO_Initure;
__HAL_RCC_GPIOF_CLK_ENABLE();
//使能GPIOF时钟//PF6
GPIO_Initure.Pin=GPIO_PIN_6;
//PF6
GPIO_Initure.Mode=GPIO_MODE_OUTPUT_PP;
//推挽输出
GPIO_Initure.Pull=GPIO_PULLUP;
//上拉
GPIO_Initure.Speed=GPIO_SPEED_FAST;
//快速
HAL_GPIO_Init(GPIOF,&GPIO_Initure);
//初始化 W25QXX_CS=1;
//SPI FLASH不选中
SPI5_Init();
//初始化SPI
SPI5_SetSpeed(SPI_BAUDRATEPRESCALER_2);
//设置为45M时钟,高速模式
W25QXX_TYPE=W25QXX_ReadID();
//读取FLASH ID.
if(W25QXX_TYPE==W25Q256)//SPI FLASH为W25Q256
{
temp=W25QXX_ReadSR(3);
//读取状态寄存器3,判断地址模式
if((temp&0X01)==0)//如果不是4字节地址模式,则进入4字节地址模式
{
W25QXX_CS=0;
//选中
SPI5_ReadWriteByte(W25X_Enable4ByteAddr);
//发送进入4字节地址模式指令
W25QXX_CS=1;
//取消片选
}
}
}//读取W25QXX的状态寄存器,W25QXX一共有3个状态寄存器
//状态寄存器1:
//BIT76543210
//SPRRVTB BP2 BP1 BP0 WEL BUSY
//SPR:默认0,状态寄存器保护位,配合WP使用
//TB,BP2,BP1,BP0:FLASH区域写保护设置
//WEL:写使能锁定
//BUSY:忙标记位(1,忙;
0,空闲)
//默认:0x00
//状态寄存器2:
//BIT76543210
//SUSCMP LB3 LB2 LB1 (R) QESRP1
//状态寄存器3:
//BIT76543210
//HOLD/RSTDRV1 DRV0 (R) (R) WPS ADP ADS
//regno:状态寄存器号,范:1~3
//返回值:状态寄存器值
u8 W25QXX_ReadSR(u8 regno)
{
u8 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;
break;
}
W25QXX_CS=0;
//使能器件
SPI5_ReadWriteByte(command);
//发送读取状态寄存器命令
byte=SPI5_ReadWriteByte(0Xff);
//读取一个字节
W25QXX_CS=1;
//取消片选
return byte;
}
//写W25QXX状态寄存器
void W25QXX_Write_SR(u8 regno,u8 sr)
{
u8 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;
//使能器件
SPI5_ReadWriteByte(command);
//发送写取状态寄存器命令
SPI5_ReadWriteByte(sr);
//写入一个字节
W25QXX_CS=1;
//取消片选
}
//W25QXX写使能
//将WEL置位
void W25QXX_Write_Enable(void)
{
W25QXX_CS=0;
//使能器件
SPI5_ReadWriteByte(W25X_WriteEnable);
//发送写使能
W25QXX_CS=1;
//取消片选
}
//W25QXX写禁止
//将WEL清零
void W25QXX_Write_Disable(void)
{
W25QXX_CS=0;
//使能器件
SPI5_ReadWriteByte(W25X_WriteDisable);
//发送写禁止指令
W25QXX_CS=1;
//取消片选
} //读取芯片ID
//返回值如下:
//0XEF13,表示芯片型号为W25Q80
//0XEF14,表示芯片型号为W25Q16
//0XEF15,表示芯片型号为W25Q32
//0XEF16,表示芯片型号为W25Q64
//0XEF17,表示芯片型号为W25Q128
//0XEF18,表示芯片型号为W25Q256
u16 W25QXX_ReadID(void)
{
u16 Temp = 0;
W25QXX_CS=0;
SPI5_ReadWriteByte(0x90);
//发送读取ID命令
SPI5_ReadWriteByte(0x00);
SPI5_ReadWriteByte(0x00);
SPI5_ReadWriteByte(0x00);
Temp|=SPI5_ReadWriteByte(0xFF)<<8;
Temp|=SPI5_ReadWriteByte(0xFF);
W25QXX_CS=1;
return Temp;
}
//读取SPI FLASH
//在指定地址开始读取指定长度的数据
//pBuffer:数据存储区
//ReadAddr:开始读取的地址(24bit)
//NumByteToRead:要读取的字节数(最大65535)
void W25QXX_Read(u8* pBuffer,u32 ReadAddr,u16 NumByteToRead)
{
u16 i;
W25QXX_CS=0;
//使能器件
SPI5_ReadWriteByte(W25X_ReadData);
//发送读取命令
if(W25QXX_TYPE==W25Q256)//如果是W25Q256的话地址为4字节的,要发送最高8位
{
SPI5_ReadWriteByte((u8)((ReadAddr)>>24));
}
SPI5_ReadWriteByte((u8)((ReadAddr)>>16));
//发送24bit地址
SPI5_ReadWriteByte((u8)((ReadAddr)>>8));
SPI5_ReadWriteByte((u8)ReadAddr);
for(i=0;
i>24));
}
SPI5_ReadWriteByte((u8)((WriteAddr)>>16));
//发送24bit地址
SPI5_ReadWriteByte((u8)((WriteAddr)>>8));
SPI5_ReadWriteByte((u8)WriteAddr);
for(i=0;
ipageremain
{
pBuffer+=pageremain;
WriteAddr+=pageremain;
NumByteToWrite-=pageremain;
//减去已经写入了的字节数
if(NumByteToWrite>256)pageremain=256;
//一次可以写入256个字节
else pageremain=NumByteToWrite;
//不够256个字节了
}
};
}
//写SPI FLASH
//在指定地址开始写入指定长度的数据
//该函数带擦除操作!
//pBuffer:数据存储区
//WriteAddr:开始写入的地址(24bit)
//NumByteToWrite:要写入的字节数(最大65535)
u8 W25QXX_BUFFER[4096];
void W25QXX_Write(u8* pBuffer,u32 WriteAddr,u16 NumByteToWrite)
{
u32 secpos;
u16 secoff;
u16 secremain;
u16 i;
u8 * 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;
i4096)secremain=4096;
//下一个扇区还是写不完
else secremain=NumByteToWrite;
//下一个扇区可以写完了
}
};
}
//擦除整个芯片
//等待时间超长...
void W25QXX_Erase_Chip(void)
{
W25QXX_Write_Enable();
//SET WEL
W25QXX_Wait_Busy();
W25QXX_CS=0;
//使能器件
SPI5_ReadWriteByte(W25X_ChipErase);
//发送片擦除命令
W25QXX_CS=1;
//取消片选
W25QXX_Wait_Busy();
//等待芯片擦除结束
}
//擦除一个扇区
//Dst_Addr:扇区地址 根据实际容量设置
//擦除一个扇区的最少时间:150ms
void W25QXX_Erase_Sector(u32 Dst_Addr)
{
//监视falsh擦除情况,测试用
//printf("fe:%x\r\n",Dst_Addr);
Dst_Addr*=4096;
W25QXX_Write_Enable();
//SET WEL
W25QXX_Wait_Busy();
W25QXX_CS=0;
//使能器件
SPI5_ReadWriteByte(W25X_SectorErase);
//发送扇区擦除指令
if(W25QXX_TYPE==W25Q256)//如果是W25Q256的话地址为4字节的,要发送最高8位
{
SPI5_ReadWriteByte((u8)((Dst_Addr)>>24));
}
SPI5_ReadWriteByte((u8)((Dst_Addr)>>16));
//发送24bit地址
SPI5_ReadWriteByte((u8)((Dst_Addr)>>8));
SPI5_ReadWriteByte((u8)Dst_Addr);
W25QXX_CS=1;
//取消片选
W25QXX_Wait_Busy();
//等待擦除完成
}
//等待空闲
void W25QXX_Wait_Busy(void)
{
while((W25QXX_ReadSR(1)&0x01)==0x01);
// 等待BUSY位清空
}
//进入掉电模式
void W25QXX_PowerDown(void)
{
W25QXX_CS=0;
//使能器件
SPI5_ReadWriteByte(W25X_PowerDown);
//发送掉电命令
W25QXX_CS=1;
//取消片选
delay_us(3);
//等待TPD
}
//唤醒
void W25QXX_WAKEUP(void)
{
W25QXX_CS=0;
//使能器件
SPI5_ReadWriteByte(W25X_ReleasePowerDown);
//send W25X_PowerDown command 0xAB
W25QXX_CS=1;
//取消片选
delay_us(3);
//等待TRES1
}