stm32 SPI-FLASH W25Q64
The W25Q64BV array is organized into 32,768 programmable pages of 256-bytes each. Up to 256 bytes can be programmed at a time. Pages can be erased in groups of 16 (sector erase), groups of 128 (32KB block erase), groups of 256 (64KB block erase) or the entire chip (chip erase)
BUSY is a read only bit in the status register (S0) that is set to a 1 state when the device is executing a Page Program, Sector Erase, Block Erase, Chip Erase or Write Status Register instruction
W25Q64
SPI配置步骤
1.使能SPI时钟
2.使能GPIO端口时钟
3.初始化GPIO,配置引脚模式
4.初始化SPI
5.使能SPI
6.SPI读写数据
7.查看SPI传输状态
举例
typedef struct
{
uint16_t SPI_Direction;
uint16_t SPI_Mode;
uint16_t SPI_DataSize; //数据帧格式
uint16_t SPI_CPOL; //时钟极性
uint16_t SPI_CPHA; //时钟相位
uint16_t SPI_NSS; //软件从设备管理
uint16_t SPI_BaudRatePrescaler; //波特率控制
uint16_t SPI_FirstBit; //帧格式
uint16_t SPI_CRCPolynomial; //CRC多项式寄存器
}SPI_InitTypeDef;
#define EN25X_WriteEnable 0x06
#define EN25X_WriteDisable 0x04
#define EN25X_ReadStatusReg 0x05
#define EN25X_WriteStatusReg 0x01
#define EN25X_ReadData 0x03
#define EN25X_FastReadData 0x0B
#define EN25X_FastReadDual 0x3B
#define EN25X_PageProgram 0x02
#define EN25X_BlockErase 0xD8
#define EN25X_SectorErase 0x20
#define EN25X_ChipErase 0xC7
#define EN25X_PowerDown 0xB9
#define EN25X_Continue_Read 0xFF
#define EN25X_ReleasePowerDown 0xAB
#define EN25X_DeviceID 0xAB
#define EN25X_ManufactDeviceID 0x90
#define EN25X_JedecDeviceID 0x9F
void SPI_init()
{
GPIO_InitTypeDef gpio =
{
GPIO_Pin_13 | GPIO_Pin_14 | GPIO_Pin_15,
GPIO_Speed_50MHz,
GPIO_Mode_AF_PP
};
SPI_InitTypeDef spi =
{
SPI_Direction_2Lines_FullDuplex,
SPI_Mode_Master, //0x0104
SPI_DataSize_8b,
SPI_CPOL_High,
SPI_CPHA_2Edge,
SPI_NSS_Soft,
SPI_BaudRatePrescaler_256,
SPI_FirstBit_MSB,
7
};
RCC_APB1PeriphClockCmd(RCC_APB1Periph_SPI2, ENABLE); //使能SPI时钟
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE); //使能GPIO端口时钟
GPIO_Init(GPIOB, &gpio); //初始化GPIO,配置引脚模式
SPI_Init(SPI2, &spi); //初始化SPI
SPI_Cmd(SPI2, ENABLE); //使能SPI
}
void SPI2_SetSpeed(u8 rate)
{
SPI2->CR1 &= 0xFFC7;
SPI2->CR1 |= rate;
SPI_Cmd(SPI2, ENABLE);
}
void EN25Q64_init()
{
GPIO_InitTypeDef gpiob =
{
GPIO_Pin_12,
GPIO_Speed_50MHz,
GPIO_Mode_Out_PP
};
GPIO_InitTypeDef gpiog13 =
{
GPIO_Pin_13,
GPIO_Speed_50MHz,
GPIO_Mode_Out_PP
};
GPIO_InitTypeDef gpiog14 =
{
GPIO_Pin_13,
GPIO_Speed_50MHz,
GPIO_Mode_Out_PP
};
GPIO_InitTypeDef gpiof =
{
GPIO_Pin_9,
GPIO_Speed_50MHz,
GPIO_Mode_Out_PP
};
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB | RCC_APB2Periph_GPIOG | RCC_APB2Periph_GPIOF, ENABLE);
GPIO_Init(GPIOB, &gpiob);
SPI2_NSS = 1; //以太网模块片选(拉高,防止干扰)
GPIO_Init(GPIOG, &gpiog13);
FLASH_CS = 1; //FLASH片选
GPIO_Init(GPIOG, &gpiog14);
SD_CS = 1; //SD卡模块片选
GPIO_Init(GPIOF, &gpiof);
TUB_4 = 1; //NRF24L01模块片选
SPI_init();
}
u8 SPI_read_write(u16 d) //SPI读写数据
{
while(SPI_I2S_GetFlagStatus(SPI2, SPI_I2S_FLAG_TXE) != SET);
SPI_I2S_SendData(SPI2, d);
while(SPI_I2S_GetFlagStatus(SPI2, SPI_I2S_FLAG_RXNE) != SET);
return SPI_I2S_ReceiveData(SPI2);
}
u16 EN25Q64_read_id()
{
u16 r;
FLASH_CS = 0;
SPI_read_write(EN25X_ManufactDeviceID);
SPI_read_write(0); //dummy
SPI_read_write(0);
SPI_read_write(0);
r = SPI_read_write(EN25X_Continue_Read) << 8;
r = SPI_read_write(EN25X_Continue_Read);
FLASH_CS = 1;
return r;
}
void EN25Q64_read_unique_id(u8 *id)
{
int i = 0;
FLASH_CS = 0;
SPI_read_write(0x4B);
SPI_read_write(0); //dummy
SPI_read_write(0);
SPI_read_write(0);
SPI_read_write(0);
for(i = 0; i < 8; i++)
{
id[i] = SPI_read_write(0xFF);
}
FLASH_CS = 1;
}
u8 EN25Q64_read_status()
{
u8 r;
FLASH_CS = 0;
SPI_read_write(EN25X_ReadStatusReg);
r = SPI_read_write(EN25X_Continue_Read);
FLASH_CS = 1;
return r;
}
void EN25Q64_read(u8 *buf, u32 addr, u16 num)
{
u16 i = 0;
FLASH_CS = 0;
SPI_read_write(EN25X_ReadData);
SPI_read_write((u8)(addr) >> 16); //24bit地址
SPI_read_write((u8)(addr) >> 8);
SPI_read_write((u8)addr);
for(i = 0; i < num; i++)
{
buf[i] = SPI_read_write(EN25X_Continue_Read);
}
FLASH_CS = 1;
}
void EN25Q64_write_enable()
{
FLASH_CS = 0;
SPI_read_write(EN25X_WriteEnable);
FLASH_CS = 1;
}
void EN25Q64_write_disable()
{
FLASH_CS = 0;
SPI_read_write(EN25X_WriteEnable);
FLASH_CS = 1;
}
void EN25Q64_write_status(u8 s)
{
FLASH_CS = 0;
SPI_read_write(EN25X_WriteDisable);
SPI_read_write(s);
FLASH_CS = 1;
}
void EN25Q64_is_busy()
{
while((EN25Q64_read_status() & 0x01) == 0x01); //BUSY
}
void EN25Q64_write_page(u8 *buf, u32 addr, u16 num) //页写
{
u16 i = 0;
EN25Q64_write_enable();
FLASH_CS = 0;
SPI_read_write(EN25X_PageProgram); //Page Program
SPI_read_write(addr >> 16);
SPI_read_write(addr >> 8);
SPI_read_write(addr);
for(i = 0; i < num; i++)
{
SPI_read_write(buf[i]);
}
FLASH_CS = 1;
EN25Q64_is_busy();
}
void EN25Q64_write_nocheck(u8 *buf, u32 addr, u16 num)
{
u8 *b = buf;
u32 a = addr;
u16 n = num;
n = 256 - (num % 256); //写满一页
while(1)
{
EN25Q64_write_page(b, a, n);
if(n == num)
{
break;
}
b += n;
a += n;
num -= n;
if(num > 256)
{
n = 256;
}
else
{
n = num;
}
}
}
void EN25Q64_sector_erase(u32 a)
{
EN25Q64_write_enable();
EN25Q64_is_busy();
FLASH_CS = 0;
SPI_read_write(EN25X_SectorErase);
SPI_read_write((u8)a >> 16);
SPI_read_write((u8)a >> 8);
SPI_read_write(a);
FLASH_CS = 1;
EN25Q64_is_busy();
}
u8 EN25QXX_BUF[4096];
void EN25Q64_write(u8 *buf, u32 addr, u16 num)
{
u16 i;
u32 pos;
u16 offset;
u16 n;
u8 *b = buf;
pos = addr / 4096; //扇区位置
offset = addr % 4096;
n = 4096 - offset ; //写满扇区
if(num <= n)
{
n = num;
}
while(1)
{
EN25Q64_read(EN25QXX_BUF, pos * 4096, 4096); //读出整个扇区
EN25Q64_sector_erase(pos); //擦除整个扇区
for(i = 0; i < n; i++)
{
EN25QXX_BUF[i + offset] = b[i];
}
EN25Q64_write_nocheck(EN25QXX_BUF, pos * 4096, 4096); //写入整个扇区
if(n == num)
{
break;
}
pos++;
num -= n;
offset = 0;
b += n;
if(num > 4096)
{
n = 4096;
}
else
{
n = num;
}
}
}
void EN25Q64_chip_erase()
{
EN25Q64_write_enable();
EN25Q64_is_busy();
FLASH_CS = 0;
SPI_read_write(EN25X_ChipErase);
FLASH_CS = 1;
EN25Q64_is_busy();
}
void EN25Q64_power_down()
{
FLASH_CS = 0;
SPI_read_write(EN25X_PowerDown);
FLASH_CS = 1;
delay_us(3);
}
void EN25Q64_release_power_down()
{
FLASH_CS = 0;
SPI_read_write(EN25X_ReleasePowerDown);
FLASH_CS = 1;
delay_us(3);
}
stm32 SPI-FLASH W25Q64的更多相关文章
- Jlink使用技巧之烧写SPI Flash存储芯片
前言 大多数玩单片机的人都知道Jlink可以烧写Hex文件,作为ARM仿真调试器,但是知道能烧写SPI Flash的人应该不多,本篇文章将介绍如何使用JLink来烧写或者读取SPI Flash存储器, ...
- Nand Flash,Nor Flash,CFI Flash,SPI Flash 之间的关系
前言: 在嵌入式开发中,如uboot的移植,kernel的移植都需要对Flash 有基本的了解.下面细说一下标题中的中Flash中的关系 一,Flash的内存存储结构 flash按照内部存 ...
- 关于STM32的FLASH操作【转载】
说到STM32的FLSAH,我们的第一反应是用来装程序的,实际上,STM32的片内FLASH不仅用来装程序,还用来装芯片配置.芯片ID.自举程序等等.当然, FLASH还可以用来装数据. FLASH分 ...
- 【iCore、iCore2 双核心板】EPCS 实验(SPI Flash)(基于Verilog语言)
_____________________________________ 深入交流QQ群: A: 204255896(1000人超级群,可加入) B: 165201798(500人超级群,满员) C ...
- nand flash,nor flash,spi flash,片上RAM,片外RAM
Flash有掉电数据保存的特点,RAM掉电则数据丢失,但是RAM的速度更高,擦写次数理论上没有限制,而Flash则不行. Nand Flash相比其余的几种flash优势在于可擦写次数多,擦写速度快, ...
- 27.some company's Spi Flash chip replace altera epcsxxx
由于altera公司的epcsxxx芯片比较贵,所以一般用其它公司的spi flash芯片代替也可以.据AlteraFAE描述:“EPCS器件也是选用某家公司的SPIFlash,只是中间经过Alter ...
- OpenRisc-32-ORPSoC烧写外部spi flash
引言 经过前面的分析和介绍,我们对ORPSoC的启动过程(http://blog.csdn.net/rill_zhen/article/details/8855743)和 ORpSoC的debug子系 ...
- SPI FLASH与NOR FLASH的区别?
1.SPI Flash (即SPI Nor Flash)是Nor Flash的一种: 2.NOR Flash根据数据传输的位数可以分为并行(Parallel)NOR Flash和串行(SPI)NOR ...
- SPI Flash(W25Q16DV) 驱动
大体上可分为以下几个部分: 1.注册设备驱动 spi_register_driver 2.分配 mtd_info 结构体 3.配置 mtd_info 结构体 4.注册 mtd_info 结构体 构建 ...
- SPI Flash(W25Q16DV) 基本操作
读取厂家\设备 ID 发送 90H 指令,再发送 00h 的地址,然后接收即可. 代码如下: void SPIFlashReadID(int *pMID, int *pDID) { SPIFlash_ ...
随机推荐
- 获取IFC构件的位置数据、方向数据
获取IFC构件的位置数据.方向数据 std::map<int, shared_ptr<BuildingEntity>> map_buildingEntity = b_model ...
- 使用Commons math做数值计算
使用Commons math做数值计算 觉得有用的话,欢迎一起讨论相互学习~Follow Me 最近使用jmetal做多目标的时候,想用一些简单的方法求最大值最小值方差和协方差矩阵,但是原生代码真的是 ...
- Ubuntu18.04 instsall XMind_8 and crack
1.dowload XMind_8 linux install zip wget https://www.xmind.cn/xmind/downloads/xmind-8-update8-linux. ...
- mybatis typeHandler类型转换器
typeHandler类型转换器 在JDBC中,需要在PreparedStatement对象中设置那些已经预编译过的SQL语句的参数.执行SQL后,会通过ResultSet对象获取得到数据库的数据,而 ...
- LODOP计算结果值的合计
LODOP中可以通过表达式等方式,计算结果并显示在LODOP预览打印的表格里,计算的是table中本来存在的数据,可以用tindex指定合计哪一列(相关博文:)那一列的数值是table中实际存在的,如 ...
- SpringBoot学习笔记:Redis缓存
SpringBoot学习笔记:Redis缓存 关于Redis Redis是一个使用ANSI C语言编写的免费开源.支持网络.可基于内存亦可以持久化的日志型.键值数据库.其支持多种存储类型,包括Stri ...
- mycat搭建环境
macos完全卸载mysql: https://blog.csdn.net/u012721519/article/details/55002626 踩过的坑: mycat1.6不支持单库分表; 最少要 ...
- Ubuntu与Window双系统安装的注意事项
UBUNTU与WINDOW双系统安装的注意事项 Ubuntu与Window双系统安装的注意事项 由 匿名 (未验证) 提交于 2019-05-18 10:07:41 登录 发表评论 29 次浏览 ...
- 小程序JS框架
- C# RESTful API
C# RESTful API REST 全称是 Representational State Transfer,有人说它是一种风格,并非一种标准,个人觉得挺有道理.它本身并没有创造新的技术.组件与服务 ...