【STM32CubeMX学习】SPI读写W25Q16

这篇具有很好参考价值的文章主要介绍了【STM32CubeMX学习】SPI读写W25Q16。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

1、SPI总线

        SPI分为主从工作模式,通常有一个主设备和一个或多个从设备,本文中MCU为主机,W25Q16为从机。

SPI通信有以下四根线:

MISO:主设备数据输入,从设备数据输出。
MOSI:主设备数据输出,从设备数据输入。
SCLK:时钟信号,由主设备产生。
CS:从设备片选信号,由主设备控制,低电平为选中。

        SPI可以同时发出和接收串行数据,主机发送一个数据的同时从机也将自己数据返回给主机。这样,双方的数据就被交换了。主机控制外设时,写操作和读操作是同步完成的。如果只进行写操作,主机只需忽略接收到的字节;反之,若主机要读取从机的一个字节,就必须发送一个空字节来引发从机的传输。

SPI的特点:高位先发送,共有四种工作模式。

CPOL(时钟极性):规定了SCK时钟信号空闲状态的电平(0-低电平,1-高电平)
CPHA(时钟相位):规定了数据是在SCK时钟的上升沿还是下降沿被采样(0-第一个时钟边沿开始采样,1-第二个时钟边沿开始采样)

模式0:CPOL=0,CPHA =0  SCK空闲为低电平,数据在SCK的上升沿被采样(提取数据)
模式1:CPOL=0,CPHA =1  SCK空闲为低电平,数据在SCK的下降沿被采样(提取数据)
模式2:CPOL=1,CPHA =0  SCK空闲为高电平,数据在SCK的下降沿被采样(提取数据)
模式3:CPOL=1,CPHA =1  SCK空闲为高电平,数据在SCK的上升沿被采样(提取数据)

2、使用STM32CubeMX配置相关引脚

【STM32CubeMX学习】SPI读写W25Q16

【STM32CubeMX学习】SPI读写W25Q16 

3、实现软件模拟SPI的四种模式

spi.h

#ifndef __SPI_H
#define __SPI_H

#include "main.h"

#define MOSI_H  HAL_GPIO_WritePin(MOSI_GPIO_Port, MOSI_Pin, GPIO_PIN_SET)  
#define MOSI_L  HAL_GPIO_WritePin(MOSI_GPIO_Port, MOSI_Pin, GPIO_PIN_RESET)  
#define SCK_H   HAL_GPIO_WritePin(SCLK_GPIO_Port, SCLK_Pin, GPIO_PIN_SET)  
#define SCK_L   HAL_GPIO_WritePin(SCLK_GPIO_Port, SCLK_Pin, GPIO_PIN_RESET)  
#define MISO    HAL_GPIO_ReadPin(MISO_GPIO_Port, MISO_Pin) 
#define F_CS_H   HAL_GPIO_WritePin(F_CS_GPIO_Port, F_CS_Pin, GPIO_PIN_SET)  
#define F_CS_L   HAL_GPIO_WritePin(F_CS_GPIO_Port, F_CS_Pin, GPIO_PIN_RESET) 

uint8_t SOFT_SPI_RW_MODE0(uint8_t write_dat);
uint8_t SOFT_SPI_RW_MODE1(uint8_t write_dat);
uint8_t SOFT_SPI_RW_MODE2(uint8_t write_dat);
uint8_t SOFT_SPI_RW_MODE3(uint8_t write_dat);

uint8_t SPI2_ReadWriteByte(uint8_t TxData);

#endif

 spi.c

#include "spi.h"

/*spi延时函数,微秒*/
static void spi_delay(uint16_t time)
{    
   uint16_t i=0;  
   while(time--)
   {
      i=10;  
      while(i--) ;    
   }
}

//CPOL:规定了SCK时钟信号空闲状态的电平(0-低电平,1-高电平)
//CPHA:规定了数据是在SCK时钟的上升沿还是下降沿被采样(0-第一个时钟边沿开始采样,1-第二个时钟边沿开始采样)

//模式0:CPOL=0,CPHA =0  SCK空闲为低电平,数据在SCK的上升沿被采样(提取数据)
//模式1:CPOL=0,CPHA =1  SCK空闲为低电平,数据在SCK的下降沿被采样(提取数据)
//模式2:CPOL=1,CPHA =0  SCK空闲为高电平,数据在SCK的下降沿被采样(提取数据)
//模式3:CPOL=1,CPHA =1  SCK空闲为高电平,数据在SCK的上升沿被采样(提取数据)

/* CPOL = 0, CPHA = 0 */
uint8_t SOFT_SPI_RW_MODE0(uint8_t write_dat)
{
  uint8_t i,read_dat = 0;
  SCK_L; 
  for(i=0;i<8;i++)
  {
    if(write_dat&0x80)
      MOSI_H;  
    else                    
      MOSI_L;  
    write_dat <<= 1;
    spi_delay(1);	
    SCK_H; 
    read_dat <<= 1;  
    
    if(MISO) 
      read_dat++; 
    spi_delay(1);
    SCK_L; 
    __nop();
  }	
  return read_dat;
}
 
/* CPOL=0,CPHA=1 */
uint8_t SOFT_SPI_RW_MODE1(uint8_t write_dat) 
{
  uint8_t i,read_dat = 0;
  SCK_L;
	for(i=0;i<8;i++)
	{
		SCK_H;     
		if(write_dat&0x80)
			MOSI_H; 
		else      
			MOSI_L;
		write_dat <<= 1;
		spi_delay(1);
		SCK_L; 
		read_dat <<= 1;  
 
		if(MISO)
			read_dat++;  
		spi_delay(1);
	}
	return read_dat;   
}
 
/* CPOL=1,CPHA=0 */
uint8_t SOFT_SPI_RW_MODE2(uint8_t write_dat) 
{
  uint8_t i,read_dat = 0; 
  SCK_H; 
	for(i=0;i<8;i++)  
	{
		if(write_dat&0x80)
			MOSI_H; 
		else      
			MOSI_L;   
		write_dat <<= 1;   
		spi_delay(1);
		SCK_L;    
		read_dat <<= 1;  
 
		if(MISO)
			read_dat++;   
		spi_delay(1);
		SCK_H; 
	}
	return read_dat;
}
 
/* CPOL = 1, CPHA = 1 */
uint8_t SOFT_SPI_RW_MODE3(uint8_t write_dat)
{
  uint8_t i,read_dat = 0;
  SCK_H; 
  for(i=0;i<8;i++)
  {
    SCK_L; 
    if(write_dat&0x80)
      MOSI_H;  
    else                    
      MOSI_L;  
    write_dat <<= 1;
    spi_delay(1);	
    SCK_H; 
    read_dat <<= 1;  
    
    if(MISO) 
      read_dat++; 
    spi_delay(1);
    __nop();
  }
  return read_dat;
}

//SPI2 读写一个字节
//TxData:要写入的字节
//返回值:读取到的字节
uint8_t SPI2_ReadWriteByte(uint8_t TxData)
{
  uint8_t Rxdata;
  Rxdata = SOFT_SPI_RW_MODE3(TxData);//使用模式3
  return Rxdata;		
}

4、实现W25Q16的读写函数

参考正点原子例程

w25q16.h

#ifndef __W25Q16_H
#define __W25Q16_H	

#include "main.h" 

extern uint8_t W25QXX_BUFFER[4096];

//W25X16读写指令表
#define W25X_WriteEnable		0x06 
#define W25X_WriteDisable		0x04 
#define W25X_ReadStatusReg		0x05 
#define W25X_WriteStatusReg		0x01 
#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 

uint16_t  W25QXX_ReadID(void);//读取FLASH ID

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_PowerDown(void);//进入掉电模式
void W25QXX_WAKEUP(void);//唤醒

#endif

w25q16.c

#include "w25q16.h"
#include "spi.h"

//容量为16M bit,2M byte,共有32个块,512个扇区 
//4Kbytes为一个扇区,16个扇区为1个块												   

//读取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
uint8_t W25QXX_ReadSR(void)   
{  
	uint8_t byte=0;   
	F_CS_L;//使能器件   
	SPI2_ReadWriteByte(W25X_ReadStatusReg);    //发送读取状态寄存器命令    
	byte=SPI2_ReadWriteByte(0Xff);   //读取一个字节  
	F_CS_H;//取消片选     
	return byte;   
} 

//写SPI_FLASH状态寄存器
//只有SPR,TB,BP2,BP1,BP0(bit 7,5,4,3,2)可以写!!!
void W25QXX_Write_SR(uint8_t sr)   
{   
	F_CS_L; //使能器件   
	SPI2_ReadWriteByte(W25X_WriteStatusReg);   //发送写取状态寄存器命令    
	SPI2_ReadWriteByte(sr);  //写入一个字节  
	F_CS_H;//取消片选     
}   

//等待空闲
void W25QXX_Wait_Busy(void)   
{   
	while((W25QXX_ReadSR()&0x01)==0x01);   //等待BUSY位清空
} 

//SPI_FLASH写使能	
//将WEL置位   
void W25QXX_Write_Enable(void)   
{
	F_CS_L;//使能器件   
  SPI2_ReadWriteByte(W25X_WriteEnable);   //发送写使能  
  F_CS_H;//取消片选     
} 

//SPI_FLASH写禁止	
//将WEL清零  
void W25QXX_Write_Disable(void)   
{  
	F_CS_L;//使能器件   
  SPI2_ReadWriteByte(W25X_WriteDisable);   //发送写禁止指令    
  F_CS_H;//取消片选     
} 			   

//读取芯片ID W25X16的ID:0XEF14
uint16_t W25QXX_ReadID(void)
{
	uint16_t Temp = 0;	  
	F_CS_L;   //使能器件   
	SPI2_ReadWriteByte(0x90);//发送读取ID命令	    
	SPI2_ReadWriteByte(0x00); 	    
	SPI2_ReadWriteByte(0x00); 	    
	SPI2_ReadWriteByte(0x00); 	 			   
	Temp|=SPI2_ReadWriteByte(0xFF)<<8;  
	Temp|=SPI2_ReadWriteByte(0xFF);	 
  F_CS_H;//取消片选     
	return Temp;
} 

//读取SPI FLASH  
//在指定地址开始读取指定长度的数据
//pBuffer:数据存储区
//ReadAddr:开始读取的地址(24bit)
//NumByteToRead:要读取的字节数(最大65535)
void W25QXX_Read(uint8_t* pBuffer,uint32_t ReadAddr,uint16_t NumByteToRead)   
{ 
 	uint16_t i;    												    
	F_CS_L;//使能器件   
  SPI2_ReadWriteByte(W25X_ReadData);//发送读取命令   
  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);   //循环读数  
  }
  F_CS_H;//取消片选     
}  

//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 WEL 
	F_CS_L;   //使能器件   
  SPI2_ReadWriteByte(W25X_PageProgram);    //发送写页命令   
  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]);//循环写数  
  F_CS_H;//取消片选     
	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;    

	secpos=WriteAddr/4096;//扇区地址 0~511 for w25x16
	secoff=WriteAddr%4096;//在扇区内的偏移
	secremain=4096-secoff;//扇区剩余空间大小   

	if(NumByteToWrite<=secremain)secremain=NumByteToWrite;//不大于4096个字节
	while(1) 
	{	
		W25QXX_Read(W25QXX_BUFFER,secpos*4096,4096);//读出整个扇区的内容
		for(i=0;i<secremain;i++)//校验数据
		{
			if(W25QXX_BUFFER[secoff+i]!=0XFF)break;//需要擦除  	  
		}
		if(i<secremain)//需要擦除
		{
			W25QXX_Erase_Sector(secpos);//擦除这个扇区
			for(i=0;i<secremain;i++)//复制
			{
				W25QXX_BUFFER[i+secoff]=pBuffer[i];	  
			}
			W25QXX_Write_NoCheck(W25QXX_BUFFER,secpos*4096,4096);//写入整个扇区  

		}
    else W25QXX_Write_NoCheck(pBuffer,WriteAddr,secremain);//写已经擦除了的,直接写入扇区剩余区间. 				   
		if(NumByteToWrite==secremain)break;//写入结束了
		else//写入未结束
		{
			secpos++;//扇区地址增1
			secoff=0;//偏移位置为0 	 

	   	pBuffer+=secremain;  //指针偏移
			WriteAddr+=secremain;//写地址偏移	   
	   	NumByteToWrite-=secremain;//字节数递减
			if(NumByteToWrite>4096)secremain=4096;//下一个扇区还是写不完
			else secremain=NumByteToWrite;//下一个扇区可以写完了
		}	 
	};	 	 
}

//擦除整个芯片
//整片擦除时间:
//W25X16:25s 
//W25X32:40s 
//W25X64:40s 
//等待时间超长...
void W25QXX_Erase_Chip(void)   
{                                             
  W25QXX_Write_Enable();  //SET WEL 
  W25QXX_Wait_Busy();   
  F_CS_L;//使能器件   
  SPI2_ReadWriteByte(W25X_ChipErase);   //发送片擦除命令  
  F_CS_H;//取消片选     
	W25QXX_Wait_Busy();//等待芯片擦除结束
}   

//擦除一个扇区
//Dst_Addr:扇区地址 0~511 for w25x16
//擦除一个扇区的最少时间:150ms
void W25QXX_Erase_Sector(uint32_t Dst_Addr)   
{   
	Dst_Addr*=4096;
  W25QXX_Write_Enable();  //SET WEL 	 
  W25QXX_Wait_Busy();   
  F_CS_L;//使能器件   
  SPI2_ReadWriteByte(W25X_SectorErase);   //发送扇区擦除指令 
  SPI2_ReadWriteByte((uint8_t)((Dst_Addr)>>16));  //发送24bit地址    
  SPI2_ReadWriteByte((uint8_t)((Dst_Addr)>>8));   
  SPI2_ReadWriteByte((uint8_t)Dst_Addr);  
  F_CS_H;//取消片选     
  W25QXX_Wait_Busy();//等待擦除完成
}   

//进入掉电模式
void W25QXX_PowerDown(void)   
{ 
  F_CS_L;//使能器件   
  SPI2_ReadWriteByte(W25X_PowerDown);   //发送掉电命令  
  F_CS_H;//取消片选     
  Delay_Us(3);//等待TPD  
}   

//唤醒
void W25QXX_WAKEUP(void)   
{  
  F_CS_L;//使能器件   
  SPI2_ReadWriteByte(W25X_ReleasePowerDown);   //send W25X_PowerDown command 0xAB    
  F_CS_H;//取消片选     
  Delay_Us(3); //等待TPD  
}   

5、验证

添加必要的变量

const uint8_t TEXT_Buffer[]={"W25Q16 TEST"};//要写入到W25Q16的字符串数组
#define SIZE sizeof(TEXT_Buffer)
uint16_t W25QXX_TYPE;//定义我们使用的flash芯片型号		
uint32_t FLASH_SIZE=2*1024*1024; //FLASH 大小为2M字节
uint8_t datatemp[SIZE];

在main函数里循环读写验证

  W25QXX_TYPE = W25QXX_ReadID();  
  printf("W25QXX_TYPE == %04x\r\n",W25QXX_TYPE);
  /* USER CODE END 2 */

  /* Infinite loop */
  /* USER CODE BEGIN WHILE */
  while (1)
  {
    /* USER CODE END WHILE */

    /* USER CODE BEGIN 3 */   
    printf("\r\nStart Write W25Q16....\r\n");
    W25QXX_Write((uint8_t*)TEXT_Buffer,FLASH_SIZE-100,SIZE); //从倒数第100个地址处开始,写入SIZE长度的数据
    printf("W25Q16 Write Finished!\r\n");//提示传送完成

    Delay_Ms(1000);
   
    printf("\r\nStart Read W25Q16....\r\n");
    W25QXX_Read(datatemp,FLASH_SIZE-100,SIZE);  //从倒数第100个地址处开始,读出SIZE个字节
    printf("The Data Readed Is:  ");//提示传送完成
    printf("%s\r\n",datatemp);//显示读到的字符串

    Delay_Ms(1000); 
  }
  /* USER CODE END 3 */

编译下载后在串口助手中查看

【STM32CubeMX学习】SPI读写W25Q16

 可以看到,芯片的ID为0xef14,循环读写也成功了。文章来源地址https://www.toymoban.com/news/detail-436220.html

到了这里,关于【STM32CubeMX学习】SPI读写W25Q16的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 【STM32学习】——SPI通信协议&SPI时序&W25Q64存储芯片&软件SPI读写

    目录 前言 一、SPI通信协议 1.概述​ 2.硬件电路  3.移位示意图 二、SPI时序 1.时序基本单元 2.完整时序波形 三、W25Q64存储芯片 1.芯片简介  2.硬件电路引脚定义  3.芯片框图 4.Flash操作注意事项 四、软件SPI读写W25Q64 五、SPI通信外设 总结 声明:学习笔记来自江科大自化协B站教

    2024年02月09日
    浏览(51)
  • 【STM32 CubeMX】SPI_Flash_W25Q64的操作方法

    在嵌入式系统开发中,使用外部 SPI Flash 存储器可以为 STM32 微控制器提供额外的存储空间,以存储程序代码、配置数据等。W25Q64 是一款常见的 SPI Flash 存储器,具有64Mb容量和SPI接口。本文将介绍如何使用 STM32 CubeMX 结合 SPI 库与 W25Q64 SPI Flash 进行集成,以便在 STM32 微控制器上

    2024年02月22日
    浏览(37)
  • 【STM32 CubeMX】SPI W25Q64功能实现

    SPI Flash 存储器在嵌入式系统中扮演着重要角色,它可以为微控制器提供额外的存储空间,并且具有快速的读写速度和较大的存储容量。W25Q64 是一款常见的 SPI Flash 存储器,容量为64Mb,采用 SPI 接口进行通信。在 STM32 微控制器上实现对 W25Q64 的功能使用,可以通过 STM32 CubeMX 和

    2024年02月22日
    浏览(62)
  • stm32(SPI读写W25Q18)

    SPI是串行外设接口(Serial Peripheral Interface)的缩写,是一种 高速的,全双工,同步 的通信总 线,并且在芯片的管脚上只占用四根线,节约了芯片的管脚,同时为PCB的布局上节省空间,提 供方便,正是出于这种简单易用的特性,越来越多的芯片集成了这种通信协议,比如 A

    2024年02月16日
    浏览(48)
  • STM32存储左右互搏 SPI总线读写SD/MicroSD/TF卡

    SD/MicroSD/TF卡是基于FLASH的一种常见非易失存储单元,由接口协议电路和FLASH构成。市面上由不同尺寸和不同容量的卡,手机领域用的TF卡实际就是MicroSD卡,尺寸比SD卡小,而电路和协议操作则是一样。这里介绍STM32CUBEIDE开发平台HAL库SPI总线操作SD/MicroSD/TF卡的例程。 除了在硬件

    2024年04月12日
    浏览(33)
  • STM-32:SPI通信协议/W25Q64简介—软件SPI读写W25Q64

    SPI是串行外设接口(Serial Peripheral Interface)的缩写,是美国摩托罗拉公司(Motorola)最先推出的一种同步串行传输规范,也是一种单片机外设芯片串行扩展接口,是一种高速、全双工、同步通信总线,所以可以在同一时间发送和接收数据,SPI没有定义速度限制,通常能达到甚

    2024年02月16日
    浏览(36)
  • 【STM32】软件SPI读写W25Q64芯片

    目录 W25Q64模块 W25Q64芯片简介 硬件电路 W25Q64框图 Flash操作注意事项 状态寄存器 ​编辑 指令集 INSTRUCTIONS​编辑 ​编辑 SPI读写W25Q64代码 硬件接线图 MySPI.c MySPI.h W25Q64 W25Q64.c W25Q64.h W25Q64_Ins.h main.c 测试 SPI通信(W25Q64芯片简介,使用SPI读写W25Q64存储器芯片)  SPI通信文章:【

    2024年02月19日
    浏览(39)
  • STM32—SPI详解入门(使用SPI通讯读写W25Q128模块)

    目录 一、SPI是什么 二、SPI物理架构 三、SPI工作原理 四、SPI工作模式 五、SPI相关寄存器介绍 六、SPI用到的结构体与函数 1.结构体 2.函数 七、W25Q128芯片 1.W25Q128介绍 2.W25Q128存储架构 3.W25Q128常用指令 4.W25Q128状态寄存器 5.W25Q128常见操作流程 八、实验(使用SPI通讯读写W25Q128模块

    2024年02月14日
    浏览(42)
  • STM32存储左右互搏 SPI总线FATS文件读写SD/MicroSD/TF卡

    SD/MicroSD/TF卡是基于FLASH的一种常见非易失存储单元,由接口协议电路和FLASH构成。市面上由不同尺寸和不同容量的卡,手机领域用的TF卡实际就是MicroSD卡,尺寸比SD卡小,而电路和协议操作则是一样。这里介绍STM32CUBEIDE开发平台HAL库SPI总线FATS文件操作读写SD/MicroSD/TF卡的例程。

    2024年04月25日
    浏览(28)
  • STM32存储左右互搏 SPI总线读写FRAM MB85RS2M

    在中低容量存储领域,除了FLASH的使用,,还有铁电存储器FRAM的使用,相对于FLASH,FRAM写操作时不需要预擦除,所以执行写操作时可以达到更高的速度,其主要优点为没有FLASH持续写操作跨页地址需要变换的要求。相比于SRAM则具有非易失性, 因此价格方面会高一些。MB85RS2M是

    2024年01月22日
    浏览(36)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包