void SPI_Flash_Write(u8* pBuffer,u32 WriteAddr,u16 NumByteToWrite) SPI_FLASH_Write_SR(0x02);//使能状态寄存器中的写存储器 SST25V_DBSY(); SPI_FLASH_Write_SR(0x02);//使能状态寄存器中的写存储器 SST25V_DBSY(); 实验目的:将数据写入外部FLASH中,然后再读出来显示在LCD上 实验平台:基于STM32F103C8T6的彩屏开发板 FLASH:SST25VF016B flash_SST.c #include "flash_SST.h" #include "spi.h" #include "delay.h" //4Kbytes为一个Sector //16个扇区为1个Block //SST25VF016B //容量为2M字节,共有32个Block(块),512个Sector(扇区) //初始化SPI FLASH的IO口 //修改状态寄存器,允许芯片存储器被写 void SPI_Flash_Init(void) { RCC->APB2ENR|=1<<2; //PORTA时钟使能 GPIOA->CRH&=0XFFFFFF0F; GPIOA->CRH|=0X00000030; //PA9 推挽 GPIOA->ODR|=1<<9; //PA9上拉 SPIx_Init(); //初始化SPI SPI_FLASH_Write_SR(0x02); //使能状态寄存器中的写存储器 SST25V_DBSY(); } //读取SPI_FLASH的状态寄存器 //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 SPI_Flash_ReadSR(void) { u8 byte=0; SPI_FLASH_CS=0; //使能器件 SPIx_ReadWriteByte(SST25_ReadStatusReg); //发送读取状态寄存器命令 byte=SPIx_ReadWriteByte(0Xff); //读取一个字节 SPI_FLASH_CS=1; //取消片选 return byte; } //写SPI_FLASH状态寄存器 //只有SPR,TB,BP2,BP1,BP0(bit 7,5,4,3,2)可以写!!! void SPI_FLASH_Write_SR(u8 sr) { SPI_FLASH_CS=0; //片选 SPIx_ReadWriteByte(SST25_EnableWriteStatusReg); //使能写状态寄存器命令 SPI_FLASH_CS=1; //取消片选 SPI_FLASH_CS=0; //片选 SPIx_ReadWriteByte(SST25_WriteStatusReg); //发送写取状态寄存器命令 SPIx_ReadWriteByte(sr); //写入一个字节 SPI_FLASH_CS=1; //取消片选 } //SPI_FLASH写使能 //将WEL置位 void SPI_FLASH_Write_Enable(void) { SPI_FLASH_CS=0; //使能器件 SPIx_ReadWriteByte(SST25_WriteEnable); //发送写使能 SPI_FLASH_CS=1; //取消片选 } //SPI_FLASH写禁止 //将WEL清零 void SPI_FLASH_Write_Disable(void) { SPI_FLASH_CS=0; //使能器件 SPIx_ReadWriteByte(SST25_WriteDisable); //发送写禁止指令 SPI_FLASH_CS=1; //取消片选 } //读取芯片ID SST25VF016的是 0XBF41 u16 SPI_Flash_ReadID(void) { u16 Temp = 0; SPI_FLASH_CS=0; //发送读取ID命令 SPIx_ReadWriteByte(0x90); //发送24位的地址 SPIx_ReadWriteByte(0x00); SPIx_ReadWriteByte(0x00); SPIx_ReadWriteByte(0x00); //读取返回的16位值 Temp=SPIx_ReadWriteByte(0xFF)<<8; //高8位数据 Temp+=SPIx_ReadWriteByte(0xFF); //底八位数据 SPI_FLASH_CS=1; return Temp; } //读取SPI FLASH //在指定地址开始读取指定长度的数据 //pBuffer:数据存储区 //ReadAddr:开始读取的地址(24bit) //NumByteToRead:要读取的字节数(最大65535即64k) void SPI_Flash_Read(u8* pBuffer,u32 ReadAddr,u16 NumByteToRead) { u16 i; SPI_FLASH_CS=0; //使能器件 SPIx_ReadWriteByte(SST25_ReadData); //发送读取命令 //发送24bit地址 SPIx_ReadWriteByte((u8)((ReadAddr)>>16)); SPIx_ReadWriteByte((u8)((ReadAddr)>>8)); SPIx_ReadWriteByte((u8)ReadAddr); for(i=0;i<NumByteToRead;i++) { pBuffer=SPIx_ReadWriteByte(0XFF); //循环读数 } SPI_FLASH_CS=1; //取消片选 } //地址自动增加的写数据A void AutoAddressIncrement_WordProgramA(u8 Byte1, u8 Byte2, u32 Addr) { SPI_FLASH_Write_Enable(); SPI_FLASH_CS=0; SPIx_ReadWriteByte(SST25_AAI_WordProgram); //输入所要写数据的起始地址 SPIx_ReadWriteByte((Addr & 0xFF0000) >> 16); SPIx_ReadWriteByte((Addr & 0xFF00) >> 8); SPIx_ReadWriteByte(Addr & 0xFF); //发送最初的两个数据 SPIx_ReadWriteByte(Byte1); SPIx_ReadWriteByte(Byte2); SPI_FLASH_CS=1; SPI_Flash_Wait_Busy(); } //地址自动增加的写数据B void AutoAddressIncrement_WordProgramB(u8 state,u8 Byte1, u8 Byte2) { SPI_FLASH_Write_Enable(); SPI_FLASH_CS=0; SPIx_ReadWriteByte(SST25_AAI_WordProgram); SPIx_ReadWriteByte(Byte1); SPIx_ReadWriteByte(Byte2); SPI_FLASH_CS=1; SPI_Flash_Wait_Busy(); if(state==1) { SPI_FLASH_Write_Disable(); } SPI_Flash_Wait_Busy(); } //结合AB构成的地址自动增加的连续数据的写入 //具有先擦除待写区域的功能 //pBuffer:为待写数据组 //WriteAddr:所写数据的起始地址 //NumByteToWrite:所要写的数据的长度 void SPI_Flash_Write(u8* pBuffer,u32 WriteAddr,u16 NumByteToWrite) { u16 i,temp; u32 secpos; u16 secoff; u16 secremain; //以下代码为擦除待写区域的代码 secpos=WriteAddr/4096; //扇区(4K)地址0~511 for SST25VF016 secoff=WriteAddr@96; //在扇区内的偏移 secremain=4096-secoff; //扇区剩余空间大小 if(NumByteToWrite<secremain) //剩余空间大于所存数据 { temp=1; } else //剩余空间小于所存数据 { i=NumByteToWrite-secremain; //判断还占了几个扇区 if(i%96==0) temp=i/4096+1; else temp=i/4096+2; } for(i=0;i<temp;i++) { SPI_Flash_Erase_Sector((secpos+i)*4096); //擦除将要写入数据的扇区 } //以下代码为将数据写入指定地址的代码 if(NumByteToWrite%2==0) { temp=NumByteToWrite/2-1; } else { temp=NumByteToWrite/2; } AutoAddressIncrement_WordProgramA(pBuffer[0], pBuffer[1],WriteAddr ); //开始写数据 for(i=1;i<temp;i++) { AutoAddressIncrement_WordProgramB(0,pBuffer[2*i], pBuffer[2*i+1]); } if(NumByteToWrite%2==0) { AutoAddressIncrement_WordProgramB(1,pBuffer[NumByteToWrite-2], pBuffer[NumByteToWrite-1]); //结束写数据 } else { AutoAddressIncrement_WordProgramB(1,pBuffer[NumByteToWrite-1],0); //结束写数据 } //写入1Byte数据 //pBuffer:待写的数据 //WriteAddr:待写数据的地址 void Flash_WriteByte(u8* pBuffer,u32 WriteAddr) { u32 secpos; secpos=WriteAddr/4096; //扇区地址 0~511 for w25x16 4096=4k SPI_Flash_Erase_Sector(secpos); //擦除这个扇区 SPI_FLASH_Write_Enable(); //SET WEL SPI_FLASH_CS=0; //使能器件 SPIx_ReadWriteByte(SST25_ByteProgram ); //发送写页命令 //发送24bit地址 SPIx_ReadWriteByte((u8)((WriteAddr)>>16)); SPIx_ReadWriteByte((u8)((WriteAddr)>>8)); SPIx_ReadWriteByte((u8)WriteAddr); //发送待写的数据 SPIx_ReadWriteByte(pBuffer[0]); SPI_FLASH_CS=1; SPI_Flash_Wait_Busy(); //等待写完成 } //擦除整个芯片 //整片擦除时间: //W25X16:25s //W25X32:40s //W25X64:40s //等待时间超长... void SPI_Flash_Erase_Chip(void) { SPI_FLASH_Write_Enable(); //SET WEL SPI_Flash_Wait_Busy(); SPI_FLASH_CS=0; //使能器件 SPIx_ReadWriteByte(SST25_ChipErase); //发送片擦除命令 SPI_FLASH_CS=1; //取消片选 SPI_Flash_Wait_Busy(); //等待芯片擦除结束 } //擦除一个扇区 //Dst_Addr:扇区地址 0~511 for w25x16 //擦除一个山区的最少时间:150ms void SPI_Flash_Erase_Sector(u32 Dst_Addr) { SPI_FLASH_Write_Enable(); //SET WEL SPI_Flash_Wait_Busy(); SPI_FLASH_CS=0; //使能器件 SPIx_ReadWriteByte(SST25_4KByte_BlockERASE); //发送扇区擦除指令 SPIx_ReadWriteByte((u8)((Dst_Addr)>>16)); //发送24bit地址 SPIx_ReadWriteByte((u8)((Dst_Addr)>>8)); SPIx_ReadWriteByte((u8)Dst_Addr); SPI_FLASH_CS=1; //取消片选 SPI_Flash_Wait_Busy(); //等待擦除完成 } //等待空闲 void SPI_Flash_Wait_Busy(void) { while ((SPI_Flash_ReadSR()&0x01)==0x01); // 等待BUSY位清空 } void SST25V_EBSY(void) { SPI_FLASH_CS=0; SPIx_ReadWriteByte( SST25_EBSY); SPI_FLASH_CS=1; } void SST25V_DBSY(void) { SPI_FLASH_CS=0; SPIx_ReadWriteByte( SST25_DBSY); SPI_FLASH_CS=1; } Flash.h #ifndef __FLASH_SST_H #define __FLASH_SST_H #include "sys.h" #define SPI_FLASH_CS PAout(9) //选中FLASH #define FLASH_ID 0XBF41 //SST25VF016读写 //指令表 #define SST25_ReadData 0x03 #define SST25_FastReadData 0x0B #define SST25_4KByte_BlockERASE 0x20 #define SST25_32KByte_BlockErase 0x52 #define SST25_64KByte_BlockErase 0xD8 #define SST25_ChipErase 0xC7 #define SST25_ByteProgram 0x02 #define SST25_AAI_WordProgram 0xAD #define SST25_ReadStatusReg 0x05 #define SST25_EnableWriteStatusReg 0x50 #define SST25_WriteStatusReg 0x01 #define SST25_WriteEnable 0x06 #define SST25_WriteDisable 0x04 #define SST25_ManufactDeviceID 0x90 #define SST25_JedecDeviceID 0x9F #define SST25_EBSY 0x70 #define SST25_DBSY 0x80 void SPI_Flash_Init(void); u16 SPI_Flash_ReadID(void); //读取FLASH ID u8 SPI_Flash_ReadSR(void); //读取状态寄存器 void SPI_FLASH_Write_SR(u8 sr); //写状态寄存器 void SPI_FLASH_Write_Enable(void); //写使能 void SPI_FLASH_Write_Disable(void); //写保护 void SPI_Flash_Read(u8* pBuffer,u32 ReadAddr,u16 NumByteToRead); //读取flash void SPI_Flash_Erase_Chip(void); //整片擦除 void SPI_Flash_Erase_Sector(u32 Dst_Addr); //扇区擦除 void SPI_Flash_Wait_Busy(void); //等待空闲 void SST25V_EBSY(void); void SST25V_DBSY(void); void Flash_WriteByte(u8* pBuffer,u32 WriteAddr); //写入1Byte数据 void AutoAddressIncrement_WordProgramA(u8 Byte1, u8 Byte2, u32 Addr); //地址自动增加的写数据A void AutoAddressIncrement_WordProgramB(u8 state,u8 Byte1, u8 Byte2); //地址自动增加的写数据B void SPI_Flash_Write(u8 pBuffer[],u32 WriteAddr,u16 NumByteToWrite); //结合AB构成的地址自动增加的连续数据的写入 #endif main.c 主函数: #include<stm32f10x_lib.h> #include"common.h" #include"TFTLCD.h" #include"spi.h" #include"key.h" #include"flash.h" const u8 TEXT_Buffer[]={"Chen An SST25VF"}; //待写入flash的数据 #define SIZE sizeof(TEXT_Buffer) //计算待写入数据的长度 int main(void) { u8 key; u8 datatemp[SIZE]; //开辟空间用于存放从flash读回的数据 Stm32_Clock_Init(9); //系统时钟初始化 delay_init(72); //延时函数的初始化 LCD_Init(); //LCD初始化 KEY_Init(); //按键初始化 SPI_Flash_Init(); //SPI关于flash的硬件接口初始化 POINT_COLOR=RED; //设置字体颜色 while(SPI_Flash_ReadID()!=FLASH_ID) //检验flash是否存在 { LCD_ShowString(60,130,"SST25VF Check Failed!"); delay_ms(500); } LCD_ShowString(60,130,"SST25VF Ready!"); LCD_ShowString(60,150,"KEY1:Write KEY2:Read"); POINT_COLOR=BLUE; while(1) { key=KEY_Scan(); //按键扫描 if(key==1) //按键1按下,开始写数据到flash { LCD_Fill(0,170,239,319,WHITE); LCD_ShowString(60,170,"Start Write SST25V"); SPI_Flash_Write((u8*)TEXT_Buffer,1000,SIZE); //写数据 LCD_ShowString(60,170,"SST25V Write Finished"); } if(key==2) //按键2按下,开始从flash读回数据 { LCD_ShowString(60,170,"Start Read SST25V"); SPI_Flash_Read(datatemp,1000,SIZE); //读数据 LCD_ShowString(60,170,"The Data Is"); LCD_ShowString(60,190,datatemp); } } } 总结:1.开始的时候,读取FLASH的ID成功,我觉得芯片一切正常,但是写入数据后读回来的全是“满屏”,纠结了一天才发现原 来是FLASH没有进行初始化,没有写 (SPI_FLASH_Write_SR(0x02);//使能状态寄存器中的写存储器 SST25V_DBSY() ) 这两句导致数据无法写入FLASH。 2.我在写程序的时候犯了个很低级的失误,在写乘法时用了 2i 结果一直提示有错误却没发现,直到过了半个小时才反应过 该写成 2*i 3.这个SST25VF016的关键在于连续数据的写入,需要仔细研究图二,我也是参考了一个网友的思路,在他得基础(A和B)拓 展出最后的 SPI_FlashWrite 函数的,不过该函数还有很多的不确定因素,大家要结合主函数中的 SIZE 来进行思考。
相关推荐
STM32芯片的SPI驱动SST25VF016B涉及了嵌入式系统中的微控制器通信和外部存储器的交互。STM32是一款基于ARM Cortex-M内核的微控制器,而SST25VF016B是一款串行闪存芯片,提供16MB的存储容量,常用于嵌入式系统中存储...
STM32F2XX SPI Flash SST25VF016BDrv 涉及到的知识点主要集中在微控制器(MCU)STM32F2XX系列、串行外围接口(SPI)通信协议以及SST25VF016B这款SPI闪存芯片上。以下是对这些关键知识点的详细说明: 1. **STM32F2XX...
总结来说,STM32_SPI1_2__Flash_SST25VF016B.zip项目涵盖了STM32微控制器使用SPI1和SPI2接口与SST25VF016B闪存进行读写操作的核心技术。这个过程涉及SPI接口的配置、命令序列的定义、数据传输以及对SST25VF016B特定...
4. **SST25VF016B命令集**:为了读写SST25VF016B,必须熟悉它的命令集,如读状态寄存器(RDSR)、读数据(READ)、写使能(WREN)、写数据块(PAGE PROG)、擦除扇区(SE)等。发送正确的命令序列,才能正确操作闪存...
本例程包含了SST25VF016B的底层驱动函数, 演示将一段字符串写入SST25VF016B的1页中,然后读出并通过USART1传送...字符串:SPI SST25VF016B Example: This is SPI DEMO, 终端上出现这一行字,说明SST25VF016B的读写正常
它们提供了不同容量的选择,如SST25VF016B(16MB)、SST25VF032B(32MB)等,适用于需要非易失性存储的应用。这些芯片通过SPI(Serial Peripheral Interface)协议与主机通信,支持快速读取和编程操作,并具有低功耗...
《SST25VF016B大容量FLASH读写驱动详解》 SST25VF016B是一款由SST(Superior Silicon Solutions)公司生产的非易失性存储器,属于SPI串行接口的Flash芯片。这款芯片拥有16兆位(2MB)的存储容量,适用于各种嵌入式...
在STM32F103上操作SST25VF系列的FLASH,首先需要配置SPI接口。这包括设置SPI时钟、模式(主模式或从模式)、数据位宽(通常为8位)以及CPOL和CPHA参数,以匹配SST25VF的时序要求。STM32的SPI接口通常有SPI1、SPI2和...
在STM32F103VET6上实现对SST25VF016B的读写,首先需要配置STM32的SPI接口。SPI(Serial Peripheral Interface)是一种同步串行通信协议,通过主设备(Master)和从设备(Slave)之间的四根信号线(SCK、MISO、MOSI和...
### STM32单片机SPI驱动SST25V Flash驱动程序详解 #### STM32 SPI接口概述 STM32微控制器系列集成了强大的SPI(Serial Peripheral Interface)接口,允许微控制器与各种类型的外围设备进行全双工串行通信。SPI接口...
在实际应用中,开发者可以利用这些驱动函数轻松地在STM32上读写SST25VF016闪存,从而扩展系统存储能力,存储应用程序、配置数据或其他重要信息。为了确保功能的正确性和优化性能,驱动函数的设计和实现必须精确符合...
在STM32项目中,FATFS允许通过SPI连接的SST25VF016B进行文件操作,如读写文件、创建目录等。 4. **ucGUI**:ucGUI是一个用于微控制器的图形用户界面库,由micriμm公司开发。它支持多种显示设备,并提供丰富的图形...
基于STM32F2芯片HAL库,main中创建任务spi2_task,如果没有使用系统,可以把sht_task任务直接改成main函数,初始化调用spi2_init()。 实际用的SST26VF032,写入一定数据后读出。 这个是硬件SPI操作串行Flash,理论W2...
这个项目的目标是将STM32F103VET6单片机通过USB接口虚拟成一个2MB的U盘,使PC能够识别并访问STM32上的SST25VF016B存储器。实现这一功能的关键在于利用STM32的USB设备模式,该模式允许STM32作为一个USB设备连接到主机...
在提供的“SST flash sst39vf400a的读写例程.doc”文档中,应包含了具体的代码示例,展示了如何在实际项目中配置SPI接口、发送读写命令以及处理读写流程。这些示例通常包括初始化函数、读取函数和写入函数,开发者...
综上所述,51单片机读写SST39VF400A涉及到SPI通信协议的理解和编程,以及针对SST39VF400A特性的操作步骤。通过fb.c文件中的代码,我们可以学习到如何实现这一过程,从而在实际项目中运用这一技术。
在STM32F103VET6上,通过SPI或QSPI接口与SST25VF016B进行通信,可以扩展MCU的存储能力,使其能够处理更大型的应用程序或数据。 Fatfs是一个开源的文件系统驱动程序,适配于多种嵌入式平台,包括STM32。它实现了FAT...
2、代码使用KEIL 标准库开发,当前在STM32F103运行,如果是STM32F103其他型号芯片,依然适用,请自行更改KEIL芯片型号以及FLASH容量即可。 3、软件下载时,请注意keil选择项是jlink还是stlink。 4、技术v:349014857...
《SST25VF064C芯片C++源代码详解及应用》 SST25VF064C是一款高性能、低功耗的串行闪存芯片,由美国SST(Superior Silicon Solutions)公司生产。这款芯片具有64Mb(8MB)的数据存储容量,广泛应用于嵌入式系统、...
SST25VF020和SST25VF040是Silicon Storage Technology公司推出的SPI接口的串行闪存产品。它们具有多种特性,适用于不同的应用场景,尤其是在空间受限和功耗要求低的环境下。 首先,SST25VF020和SST25VF040的共同...