[硬件]关于SPI Flash那些你不知道的事儿

简介: 刚开始学习STM32时,对SPI Flash的块、扇区的概念模糊不清,现在回头再看,感觉豁然开朗

以华邦W25Q128为例,详解SPI Flash的特点,读写注意事项,和地址范围等。


和EPROM的区别

以AT24C02 EPROM和W25Q128 SPI Flash为例。


  • EPROM通常采用是IIC串行总线,低速,单双工,通信速率一般是百KHz。而SPI Flash是采用的SPI总线,高速,全双工,通讯速率一般是百MHz。SPI Flash属于Flash ROM闪存,相比于EPROM,读写速度更快


  • EPROM通常用于存储不频繁读取的数据,如配置信息等,而EPROM通常用来存储经常读取的数据,如字库文件等。


  • EPROM读写比较随意,想写那个地址写那个,想读哪个地址读哪个!而SPI Flash则比较规范,擦除的最小单位是扇区。向某个地址写入数据时, 要先读取这个地址的数据是否为0xFF,如果不是0xFF,那么这个数据写入失败。所以通常的写操作是,在写某个地址之前,直接擦除这个地址所在的那个扇区,然后再写数据。当然,如果这个扇区的所有内容都是0xFF,则无需擦除,可以直接写入。


  • EPROM通常容量比较小,大小为KB级的,如AT24C02是2KB,而SPI Flash容量比较大,大小为MB级的,如W25Q16是16Mbit,也就是2MB。


  • EPROM型号通常是xx24系列,而SPI Flash通常是xx25系列,所以从芯片型号我们也可以看出ROM类型。


  • EPROM数据保存时间大约是100年,而SPI Flash数据保存时间为20年。


  • EPROM的读写次数为100万次左右,SPI Flash读写次数为10万次左右


AT24C02读写次数和存储时间

640.png


W25Q128读写次数和存储时间

640.png


块、扇区、页傻傻分不清

以华邦的W25Q128为例,容量为128Mbits,注意这里的单位是bit,换算成字节(Byte),也就是:

128Mbits/8 = 16MB = 16*1024KB = 16384 KB = 16,777,216B,所以很容易计算出整个存储空间的地址范围:0x000000~0xFFFFFF


SPI Flash和EPROM的很大的一个不同就是多了块、扇区、页的概念。


W25Q128的整个存储空间被分成了256个块(Block),每个块包含16个扇区(Sector),每个扇区又包括16个页。


所以,如果按照块来计算,W25Q128包括256个块。


如果按照扇区来计算,W25Q128包括256*16=4096个扇区。


如果按照页来计算的话,W25Q128包括4096*16=65536个页。


每个块的大小是:16384KB/256 = 64KB


每个扇区的大小是:64KB/16 = 4KB


每个页的大小是:4KB/16 = 256B


但是实际上,我们在进行读写操作时,都是区分块和扇区,不区分页的。包括在官方的Datasheet中,并没有重点提及页的地址范围。

640.png


地址范围

从存储容量来看,我们可以轻松的计算出W25Q128的整个存储空间的地址范围:


0x000000~0xFFFFFF,也就是地址最大是24位。根据块的大小是64KB,扇区的大小是4KB,我们可以计算出每个块和扇区的地址范围:


块0的地址:`0x000000~0x00FFFF`
块1的地址:`0x010000~0x01FFFF`
.....
块255的地址:`0xFF0000~0xFFFFFF`


对于每个块,以块0为例:

块0扇区0的地址:`0x000000~0x000FFF`
块0扇区1的地址:`0x001000~0x001FFF`
....
块0扇区15的地址:`0x00F000~0x00FFF`


不知道你是否发现了,地址的高8位(23-16位)表示块的位置,第15-12位为扇区的位置。


例如,块10的第7个扇区的地址范围:0x0A 7 000 ~ 0x0A 7 FFF


W25Q128支持读取任意一个地址的数据,范围:0x000000~0xFFFFFF


根据绝对地址,获取这个地址所在的块和扇区位置就很简单了:

/* 存储地址 */
uint32_t addr = 0xC0A002;
/* 23-16位是块的位置 */
uint8_t block = addr >> 16;    /* (addr & 0xFF0000)>>16*/
/* 15-12位是扇区的位置 */
uint8_t sector = (addr << 16) >> 28; /* (addr & 0x00F000)>>12 */
uart_init(115200);
printf("addr:%x, block:%d, sector:%d\r\n", addr, block, sector);


运行结果

640.png


常用指令

W25Q128的擦除,可以通过指令配置为单独的扇区擦除,单独的块擦除,或者整片擦除,整片擦除时间会比较长。

0xC7:整片擦除
0xD8:块擦除
0x20:扇区擦除
0xAB:获取芯片ID
0x90:获取芯片型号
0x06:写使能
0x04:禁止写
0xB9:进入掉电模式,功耗极低
0xAB:退出掉电模式


发送0x90命令之后的返回值表示当前器件的型号:

/*
0XEF13,表示芯片型号为W25Q80
0XEF14,表示芯片型号为W25Q16
0XEF15,表示芯片型号为W25Q32
0XEF16,表示芯片型号为W25Q64
0XEF17,表示芯片型号为W25Q128
*/


使用Jlink烧写SPI Flash

大多数玩单片机的人都知道Jlink可以烧写Hex文件,作为ARM仿真调试器,但是知道能烧写SPI Flash的人应该不多。

JLink软件包含的工具中,有一个是JFlashSPI工具,可以烧写和读取SPI存储器。

可以参考:Jlink使用技巧之烧写SPI Flash存储芯片

目录
相关文章
|
存储 芯片 内存技术
Jlink使用技巧之读写SPI Flash存储芯片
Jlink使用技巧之读写SPI Flash存储芯片
699 0
Jlink使用技巧之读写SPI Flash存储芯片
|
11天前
NUC980修改内核支持spi-nand
NUC980修改内核支持spi-nand
20 2
|
8月前
|
存储 缓存 算法
NAND FLASH 和NOR FLASH的区别
NAND FLASH 和NOR FLASH的区别
99 0
|
XML 测试技术 网络安全
开发工具:USB转IIC/I2C/SPI/UART适配器模块可编程开发板
总的思路是通过USB或者UART接口发送一些协议字符串,由模块转换成上面几种接口的硬件时序电信号,实现与这几种接口芯片、设备的快速测试。 首先声明一下,大家都是搞硬件开发的,这几种接口当然是很简单的事,但有些时候对于一个新的设备或者芯片的测试,有个现成的工具当然更顺手,节省时间,也更可靠嘛。
|
XML 测试技术 网络安全
开发调试工具:可编程USB转IIC/I2C/SPI/UART适配器模块开发板
发个方便测试I2C、SPI、1Wire接口的工具模块 总的思路是通过USB或者UART接口发送一些协议字符串,由模块转换成上面几种接口的硬件时序电信号,实现与这几种接口芯片、设备的快速测试。
|
Linux 芯片
Linux驱动分析之SPI设备
前面我们对SPI控制器驱动进行了分析,接下来来分析SPI设备驱动。我们以DS1302驱动作为分析对象。DS1302是一款RTC芯片,估计很多人在学单片机时用到过。RTC芯片算是比较简单的,也方便分析理解。
Linux驱动分析之SPI设备
|
存储 芯片 内存技术
Jlink使用技巧之烧写SPI Flash存储芯片
Jlink使用技巧之烧写SPI Flash存储芯片
355 0
Jlink使用技巧之烧写SPI Flash存储芯片
|
XML 传感器 测试技术
开发调试工具:USB转IIC/I2C/SPI/UART适配器模块可编程开发板
发个方便测试I2C、SPI、1Wire接口的工具模块 总的思路是通过USB或者UART接口发送一些协议字符串,由模块转换成上面几种接口的硬件时序电信号,实现与这几种接口芯片、设备的快速测试。 首先声明一下,大家都是搞硬件开发的,这几种接口当然是很简单的事,但有些时候对于一个新的设备或者芯片的测试,有个现成的工具当然更顺手,节省时间,也更可靠嘛。
开发调试工具:USB转IIC/I2C/SPI/UART适配器模块可编程开发板
|
存储 芯片 异构计算
FPGA-利用SPI总线进行flash操作
FPGA-利用SPI总线进行flash操作
328 0
FPGA-利用SPI总线进行flash操作
|
存储 开发工具 芯片
ZYNQ-QSPI Flash读写操作(一)
ZYNQ-QSPI Flash读写操作
1066 0
ZYNQ-QSPI Flash读写操作(一)