【正点原子探索者STM32F407开发板例程连载+教学】第30章 SPI通信实验

2023-11-20

第三十章 SPI 实验

 

[mw_shl_code=c,true]1.硬件平台:正点原子探索者STM32F407开发板 2.软件平台:MDK5.1 3.固件库版本:V1.4.0 [/mw_shl_code]

 

 

本章我们将向大家介绍STM32F4的SPI功能。在本章中,我们将使用STM32F4自带的SPI来实现对外部FLASH(W25Q128)的读写,并将结果显示在TFTLCD模块上。本章分为如下几个部分:

30.1 SPI 简介

30.2 硬件设计

30.3 软件设计

30.4 下载验证

30.1 SPI 简介

SPI 是英语Serial Peripheral interface的缩写,顾名思义就是串行外围设备接口。是Motorola首先在其MC68HCXX系列处理器上定义的。SPI接口主要应用在 EEPROM,FLASH,实时时钟,AD转换器,还有数字信号处理器和数字信号解码器之间。SPI,是一种高速的,全双工,同步的通信总线,并且在芯片的管脚上只占用四根线,节约了芯片的管脚,同时为PCB的布局上节省空间,提供方便,正是出于这种简单易用的特性,现在越来越多的芯片集成了这种通信协议,STM32F4也有SPI接口。下面我们看看SPI的内部简明图(图30.1.1):

 

 

图30.1.1 SPI内部结构简明图

 

SPI接口一般使用4条线通信:

MISO 主设备数据输入,从设备数据输出。

MOSI 主设备数据输出,从设备数据输入。

SCLK时钟信号,由主设备产生。

CS从设备片选信号,由主设备控制。

从图中可以看出,主机和从机都有一个串行移位寄存器,主机通过向它的SPI串行寄存器写入一个字节来发起一次传输。寄存器通过MOSI信号线将字节传送给从机,从机也将自己的移位寄存器中的内容通过MISO信号线返回给主机。这样,两个移位寄存器中的内容就被交换。外设的写操作和读操作是同步完成的。如果只进行写操作,主机只需忽略接收到的字节;反之,若主机要读取从机的一个字节,就必须发送一个空字节来引发从机的传输。 

SPI主要特点有:可以同时发出和接收串行数据;可以当作主机或从机工作;提供频率可编程时钟;发送结束中断标志;写冲突保护;总线竞争保护等。

SPI总线四种工作方式 SPI 模块为了和外设进行数据交换,根据外设工作要求,其输出串行同步时钟极性和相位可以进行配置,时钟极性(CPOL)对传输协议没有重大的影响。如果 CPOL=0,串行同步时钟的空闲状态为低电平;如果CPOL=1,串行同步时钟的空闲状态为高电平。时钟相位(CPHA)能够配置用于选择两种不同的传输协议之一进行数据传输。如果CPHA=0,在串行同步时钟的第一个跳变沿(上升或下降)数据被采样;如果CPHA=1,在串行同步时钟的第二个跳变沿(上升或下降)数据被采样。SPI主模块和与之通信的外设备时钟相位和极性应该一致。

不同时钟相位下的总线数据传输时序如图30.1.2所示:

 

图30.1.2 不同时钟相位下的总线传输时序(CPHA=0/1)

STM32F4的SPI功能很强大,SPI时钟最高可以到37.5Mhz,支持DMA,可以配置为SPI协议或者I2S协议(支持全双工I2S)。

本章,我们将使用STM32F4的SPI来读取外部SPI FLASH芯片(W25Q128),实现类似上节的功能。这里对SPI我们只简单介绍一下SPI的使用,STM32F4的SPI详细介绍请参考《STM32F4xx中文参考手册》第721页,27节。然后我们再介绍下SPI  FLASH芯片。

这节,我们使用STM32F4的SPI1的主模式,下面就来看看SPI1部分的设置步骤吧。SPI相关的库函数和定义分布在文件stm32f4xx_spi.c以及头文件stm32f4xx_spi.h中。STM32的主模式配置步骤如下:

1)配置相关引脚的复用功能,使能SPI1时钟。

我们要用SPI1,第一步就要使能SPI1的时钟,SPI1的时钟通过APB2ENR的第12位来设置。其次要设置SPI1的相关引脚为复用(AF5)输出,这样才会连接到SPI1上。这里我们使用的是PB3、4、5这3个(SCK.、MISO、MOSI,CS使用软件管理方式),所以设置这三个为复用IO,复用功能为AF5。

使能SPI1时钟的方法为:

RCC_APB2PeriphClockCmd(RCC_APB2Periph_SPI1, ENABLE);//使能SPI1时钟

复用PB3,PB4,PB5为SPI1引脚的方法为:

GPIO_PinAFConfig(GPIOB,GPIO_PinSource3,GPIO_AF_SPI1); //PB3复用为 SPI1

GPIO_PinAFConfig(GPIOB,GPIO_PinSource4,GPIO_AF_SPI1); //PB4复用为 SPI1

GPIO_PinAFConfig(GPIOB,GPIO_PinSource5,GPIO_AF_SPI1); //PB5复用为 SPI1

同时我们要设置相应的引脚模式为复用功能模式:

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;//复用功能

2)初始化SPI1,设置SPI1工作模式等。

这一步全部是通过SPI1_CR1来设置,我们设置SPI1为主机模式,设置数据格式为8位,然后通过CPOL和CPHA位来设置SCK时钟极性及采样方式。并设置SPI1的时钟频率(最大37.5Mhz),以及数据的格式(MSB在前还是LSB在前)。在库函数中初始化SPI的函数为:

void SPI_Init(SPI_TypeDef* SPIx, SPI_InitTypeDef* SPI_InitStruct);

跟其他外设初始化一样,第一个参数是SPI标号,这里我们是使用的SPI1。下面我们来看看第二个参数结构体类型SPI_InitTypeDef的定义:

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;

}SPI_InitTypeDef;

结构体成员变量比较多,接下来我们简单讲解一下:

第一个参数SPI_Direction是用来设置SPI的通信方式,可以选择为半双工,全双工,以及串行发和串行收方式,这里我们选择全双工模式SPI_Direction_2Lines_FullDuplex。

第二个参数SPI_Mode用来设置SPI的主从模式,这里我们设置为主机模式SPI_Mode_Master,当然有需要你也可以选择为从机模式SPI_Mode_Slave。

第三个参数SPI_DataSiz为8位还是16位帧格式选择项,这里我们是8位传输,选择SPI_DataSize_8b。

第四个参数SPI_CPOL用来设置时钟极性,我们设置串行同步时钟的空闲状态为高电平所以我们选择SPI_CPOL_High。

第五个参数SPI_CPHA用来设置时钟相位,也就是选择在串行同步时钟的第几个跳变沿(上升或下降)数据被采样,可以为第一个或者第二个条边沿采集,这里我们选择第二个跳变沿,所以选择SPI_CPHA_2Edge

第六个参数SPI_NSS设置NSS信号由硬件(NSS管脚)还是软件控制,这里我们通过软件控制NSS关键,而不是硬件自动控制,所以选择SPI_NSS_Soft。

第七个参数SPI_BaudRatePrescaler很关键,就是设置SPI波特率预分频值也就是决定SPI的时钟的参数,从2分频到256分频8个可选值,初始化的时候我们选择256分频值SPI_BaudRatePrescaler_256, 传输速度为84M/256=328.125KHz。

第八个参数SPI_FirstBit设置数据传输顺序是MSB位在前还是LSB位在前,,这里我们选择SPI_FirstBit_MSB高位在前。

第九个参数SPI_CRCPolynomial是用来设置CRC校验多项式,提高通信可靠性,大于1即可。

设置好上面9个参数,我们就可以初始化SPI外设了。初始化的范例格式为:

SPI_InitTypeDef  SPI_InitStructure;

SPI_InitStructure.SPI_Direction = SPI_Direction_2Lines_FullDuplex;  //双线双向全双工

SPI_InitStructure.SPI_Mode = SPI_Mode_Master;            //主SPI

SPI_InitStructure.SPI_DataSize = SPI_DataSize_8b;  // SPI发送接收8位帧结构

SPI_InitStructure.SPI_CPOL = SPI_CPOL_High;//串行同步时钟的空闲状态为高电平

SPI_InitStructure.SPI_CPHA = SPI_CPHA_2Edge;//第二个跳变沿数据被采样

SPI_InitStructure.SPI_NSS = SPI_NSS_Soft;    //NSS信号由软件控制

SPI_InitStructure.SPI_BaudRatePrescaler = SPI_BaudRatePrescaler_256; //预分频256

SPI_InitStructure.SPI_FirstBit = SPI_FirstBit_MSB;  //数据传输从MSB位开始

SPI_InitStructure.SPI_CRCPolynomial = 7;       //CRC值计算的多项式

SPI_Init(SPI2, &SPI_InitStructure);  //根据指定的参数初始化外设SPIx寄存器

3)使能SPI1

这一步通过SPI1_CR1的bit6来设置,以启动SPI1,在启动之后,我们就可以开始SPI通讯了。库函数使能SPI1的方法为:

SPI_Cmd(SPI1, ENABLE); //使能SPI1外设

4)SPI传输数据

通信接口当然需要有发送数据和接受数据的函数,固件库提供的发送数据函数原型为:

void SPI_I2S_SendData(SPI_TypeDef* SPIx, uint16_t Data);

这个函数很好理解,往SPIx数据寄存器写入数据 Data,从而实现发送。

固件库提供的接受数据函数原型为:

uint16_t SPI_I2S_ReceiveData(SPI_TypeDef* SPIx) ;

这个函数也不难理解,从SPIx数据寄存器读出接受到的数据。

5)查看SPI传输状态

在SPI传输过程中,我们经常要判断数据是否传输完成,发送区是否为空等等状态,这是通过函数SPI_I2S_GetFlagStatus实现的,这个函数很简单就不详细讲解,判断发送是否完成的方法是:

SPI_I2S_GetFlagStatus(SPI1, SPI_I2S_FLAG_RXNE);

SPI1的使用就介绍到这里,接下来介绍一下W25Q128。W25Q128是华邦公司推出的大容量SPI FLASH产品,W25Q128的容量为128Mb,该系列还有W25Q80/16/32/64等。ALIENTEK所选择的W25Q128容量为128Mb,也就是16M字节。

W25Q128将16M的容量分为256个块(Block),每个块大小为64K字节,每个块又分为16个扇区(Sector),每个扇区4K个字节。W25Q128的最小擦除单位为一个扇区,也就是每次必须擦除4K个字节。这样我们需要给W25Q128开辟一个至少4K的缓存区,这样对SRAM要求比较高,要求芯片必须有4K以上SRAM才能很好的操作。

W25Q128的擦写周期多达10W次,具有20年的数据保存期限,支持电压为2.7~3.6V,W25Q128支持标准的SPI,还支持双输出/四输出的SPI,最大SPI时钟可以到80Mhz(双输出时相当于160Mhz,四输出时相当于320M),更多的W25Q128的介绍,请参考W25Q128的DATASHEET。

30.2 硬件设计

本章实验功能简介:开机的时候先检测W25Q128是否存在,然后在主循环里面检测两个按键,其中1个按键(KEY1)用来执行写入W25Q128的操作,另外一个按键(KEY0)用来执行读出操作,在TFTLCD模块上显示相关信息。同时用DS0提示程序正在运行。

所要用到的硬件资源如下:

1)  指示灯DS0

2)  KEY_UP和KEY1按键

3) TFTLCD模块

4) SPI

5) W25Q128

这里只介绍W25Q128与STM32F4的连接,板上的W25Q128是直接连在STM32F4的SPI1上的,连接关系如图30.2.1所示:

 

图30.2.1 STM32F4与W25Q128连接电路图

这里,我们的F_CS是连接在PB14上面的,另外要特别注意:W25Q128和NRF24L01共用SPI1,所以这两个器件在使用的时候,必须分时复用(通过片选控制)才行。

30.3 软件设计

打开我们光盘的SPI实验工程,可以看到我们加入了spi.c,flash.c文件以及头文件spi.h和flash.h,同时引入了库函数文件stm32f4xx_spi.c文件以及头文件stm32f4xx_spi.h。

打开spi.c文件,看到如下代码:

//以下是SPI模块的初始化代码,配置成主机模式                                             

//SPI口初始化

//这里针是对SPI1的初始化

void SPI1_Init(void)

{    

  GPIO_InitTypeDef  GPIO_InitStructure;

  SPI_InitTypeDef  SPI_InitStructure;

      

  RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA, ENABLE);//使能GPIOA时钟

  RCC_APB2PeriphClockCmd(RCC_APB2Periph_SPI1, ENABLE);// 使能SPI1时钟

 

  //GPIOFB3,4,5初始化设置: 复用功能输出  

  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_3|GPIO_Pin_4|GPIO_Pin_5;//PB3~5

  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;//复用功能

  GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;//推挽输出

  GPIO_InitStructure.GPIO_Speed = GPIO_Speed_100MHz;//100MHz

  GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP;//上拉

  GPIO_Init(GPIOB, &GPIO_InitStructure);// 初始化

 

  //配置引脚复用映射

GPIO_PinAFConfig(GPIOB,GPIO_PinSource3,GPIO_AF_SPI1); //PB3复用为 SPI1

GPIO_PinAFConfig(GPIOB,GPIO_PinSource4,GPIO_AF_SPI1); //PB4复用为 SPI1

GPIO_PinAFConfig(GPIOB,GPIO_PinSource5,GPIO_AF_SPI1); //PB5复用为 SPI1

 

//这里只针对SPI口初始化

RCC_APB2PeriphResetCmd(RCC_APB2Periph_SPI1,ENABLE);//复位SPI1

RCC_APB2PeriphResetCmd(RCC_APB2Periph_SPI1,DISABLE);//停止复位SPI1

 

SPI_InitStructure.SPI_Direction = SPI_Direction_2Lines_FullDuplex;  //设置SPI全双工

SPI_InitStructure.SPI_Mode = SPI_Mode_Master;        //设置SPI工作模式:主SPI

SPI_InitStructure.SPI_DataSize = SPI_DataSize_8b;     //设置SPI的数据大小: 8位帧结构

SPI_InitStructure.SPI_CPOL = SPI_CPOL_High;//串行同步时钟的空闲状态为高电平

SPI_InitStructure.SPI_CPHA = SPI_CPHA_2Edge; //数据捕获于第二个时钟沿

SPI_InitStructure.SPI_NSS = SPI_NSS_Soft;   //NSS信号由硬件管理

SPI_InitStructure.SPI_BaudRatePrescaler = SPI_BaudRatePrescaler_256;    //预分频256

SPI_InitStructure.SPI_FirstBit = SPI_FirstBit_MSB; //数据传输从MSB位开始

SPI_InitStructure.SPI_CRCPolynomial = 7;    //CRC值计算的多项式

SPI_Init(SPI1, &SPI_InitStructure); //根据指定的参数初始化外设SPIx寄存器

 

SPI_Cmd(SPI1, ENABLE); //使能SPI1

SPI1_ReadWriteByte(0xff);//启动传输          

}  

//SPI1速度设置函数

//SPI速度=fAPB2/分频系数

//入口参数范围:@ref SPI_BaudRate_Prescaler

//SPI_BaudRatePrescaler_2~SPI_BaudRatePrescaler_256

//fAPB2时钟一般为84Mhz:

void SPI1_SetSpeed(u8 SPI_BaudRatePrescaler)

{

   assert_param(IS_SPI_BAUDRATE_PRESCALER(SPI_BaudRatePrescaler));//判断有效性

       SPI1->CR1&=0XFFC7;//位3-5清零,用来设置波特率

       SPI1->CR1|=SPI_BaudRatePrescaler;   //设置SPI1速度

       SPI_Cmd(SPI1,ENABLE); //使能SPI1

}

//SPI1 读写一个字节

//TxData:要写入的字节

//返回值:读取到的字节

u8 SPI1_ReadWriteByte(u8 TxData)

{                                

 

  while (SPI_I2S_GetFlagStatus(SPI1, SPI_I2S_FLAG_TXE) == RESET){}//等待发送区空 

SPI_I2S_SendData(SPI1, TxData); //通过外设SPIx发送一个byte  数据

while (SPI_I2S_GetFlagStatus(SPI1, SPI_I2S_FLAG_RXNE) == RESET){} //等待接收完

return SPI_I2S_ReceiveData(SPI1); //返回通过SPIx最近接收的数据      

                

}

此部分代码主要初始化SPI,这里我们选择的是SPI1,所以在SPI1_Init函数里面,其相关的操作都是针对SPI1的,其初始化步骤和我们上面介绍的一样。在初始化之后,我们就可以开始使用SPI1了,这里特别注意,SPI初始化函数的最后有一个启动传输,这句话最大的作用就是维持MOSI为高电平,而且这句话也不是必须的,可以去掉。

在SPI1_Init函数里面,把SPI1的频率设置成了最低(84Mhz,256分频)。在外部函数里面,我们通过SPI1_SetSpeed来设置SPI1的速度,而我们的数据发送和接收则是通过SPI1_ReadWriteByte函数来实现的。

接下来我们来看看w25qxx.c文件内容。由于篇幅所限,详细代码,这里就不贴出了。我们仅介绍几个重要的函数,首先是W25QXX_Read函数,该函数用于从W25Q128的指定地址读出指定长度的数据。其代码如下:     

//读取SPI FLASH 

//在指定地址开始读取指定长度的数据

//pBuffer:数据存储区

//ReadAddr:开始读取的地址(24bit)

//NumByteToRead:要读取的字节数(最大65535)

void W25QXX_Read(u8* pBuffer,u32 ReadAddr,u16 NumByteToRead)  

{

      u16 i;                                                                       

       W25QXX_CS=0;                         //使能器件  

    SPI1_ReadWriteByte(W25X_ReadData);      //发送读取命令  

    SPI1_ReadWriteByte((u8)((ReadAddr)>>16));  //发送24bit地址   

    SPI1_ReadWriteByte((u8)((ReadAddr)>>8));  

    SPI1_ReadWriteByte((u8)ReadAddr);  

    for(i=0;i<NumByteToRead;i++)

       {

        pBuffer=SPI1_ReadWriteByte(0XFF);   //循环读数 

    }

       W25QXX_CS=1;                                               

}

由于W25Q128支持以任意地址(但是不能超过W25Q128的地址范围)开始读取数据,所以,这个代码相对来说就比较简单了,在发送24位地址之后,程序就可以开始循环读数据了,其地址会自动增加的,不过要注意,不能读的数据超过了W25Q128的地址范围哦!否则读出来的数据,就不是你想要的数据了。

有读的函数,当然就有写的函数了,接下来,我们介绍W25QXX_Write这个函数,该函数的作用与W25QXX_Flash_Read的作用类似,不过是用来写数据到W25Q128里面的,代码如下:

//写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;i<secremain;i++)//校验数据

              {

                     if(W25QXX_BUF[secoff+i]!=0XFF)break;//需要擦除       

              }

              if(i<secremain)//需要擦除

              {

                     W25QXX_Erase_Sector(secpos);//擦除这个扇区

                     for(i=0;i<secremain;i++)          //复制

                     {

                            W25QXX_BUF[i+secoff]=pBuffer;   

                     }

                     W25QXX_Write_NoCheck(W25QXX_BUF,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;               //下一个扇区可以写完了

              }    

       };

}

该函数可以在W25Q128的任意地址开始写入任意长度(必须不超过W25Q128的容量)的数据。我们这里简单介绍一下思路:先获得首地址(WriteAddr)所在的扇区,并计算在扇区内的偏移,然后判断要写入的数据长度是否超过本扇区所剩下的长度,如果不超过,再先看看是否要擦除,如果不要,则直接写入数据即可,如果要则读出整个扇区,在偏移处开始写入指定长度的数据,然后擦除这个扇区,再一次性写入。当所需要写入的数据长度超过一个扇区的长度的时候,我们先按照前面的步骤把扇区剩余部分写完,再在新扇区内执行同样的操作,如此循环,直到写入结束。这里我们还定义了一个W25QXX_BUFFER的全局变量,用于擦除时缓存扇区内的数据。

其他的代码就比较简单了,我们这里不介绍了。对于头文件w25qxx.h,这里面就定义了一些与W25Q128操作相关的命令和函数(部分省略了),这些命令在W25Q128的数据手册上都有详细的介绍,感兴趣的读者可以参考该数据手册。

最后,我们看看main函数,代码如下:

//要写入到W25Q128的字符串数组

const u8 TEXT_Buffer[]={"Explorer STM32F4 SPI TEST"};

#define SIZE sizeof(TEXT_Buffer)     

int main(void)

{           

u8 key, datatemp[SIZE];

       u16 i=0;

       u32 FLASH_SIZE;

       NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);//设置系统中断优先级分组2

       delay_init(168);  //初始化延时函数

       uart_init(115200);         //初始化串口波特率为115200    

       LED_Init();                         //初始化LED

      LCD_Init();                         //LCD初始化

       KEY_Init();                        //按键初始化 

       W25QXX_Init();                  //W25QXX初始化

      POINT_COLOR=RED;

       LCD_ShowString(30,50,200,16,16,"Explorer STM32F4");      

       LCD_ShowString(30,70,200,16,16,"SPI TEST");      

       LCD_ShowString(30,90,200,16,16,"ATOM@ALIENTEK");

       LCD_ShowString(30,110,200,16,16,"2014/5/7");      

       LCD_ShowString(30,130,200,16,16,"KEY1:Write  KEY0:Read");  //显示提示信息                   while(W25QXX_ReadID()!=W25Q128)      //检测不到W25Q128

       {

              LCD_ShowString(30,150,200,16,16,"W25Q128 Check Failed!");

              delay_ms(500);

              LCD_ShowString(30,150,200,16,16,"Please Check!        ");

              delay_ms(500);

              LED0=!LED0;              //DS0闪烁

       }

       LCD_ShowString(30,150,200,16,16,"W25Q128 Ready!");

       FLASH_SIZE=128*1024*1024;   //FLASH 大小为2M字节

      POINT_COLOR=BLUE;                     //设置字体为蓝色   

       while(1)

       {

              key=KEY_Scan(0);

              if(key==KEY1_PRES)//KEY1按下,写入W25Q128

              {

                     LCD_Fill(0,170,239,319,WHITE);//清除半屏   

                    LCD_ShowString(30,170,200,16,16,"Start Write W25Q128....");

                     W25QXX_Write((u8*)TEXT_Buffer,FLASH_SIZE-100,SIZE);             

//从倒数第100个地址处开始,写入SIZE长度的数据

                     LCD_ShowString(30,170,200,16,16,"W25Q128 Write Finished!");//提示完成

              }

              if(key==KEY0_PRES)//KEY0按下,读取字符串并显示

              {

                    LCD_ShowString(30,170,200,16,16,"Start Read W25Q128.... ");

                     W25QXX_Read(datatemp,FLASH_SIZE-100,SIZE); 

//从倒数第100个地址处开始,读出SIZE个字节

                     LCD_ShowString(30,170,200,16,16,"The Data Readed Is:   ");       //提示传送完成

                     LCD_ShowString(30,190,200,16,16,datatemp);   //显示读到的字符串

              }

              i++;

              delay_ms(10);

              if(i==20)

              {

                     LED0=!LED0;//提示系统正在运行    

                     i=0;

              }              

       }      

}

这部分代码和IIC实验那部分代码大同小异,我们就不多说了,实现的功能就和IIC差不多,不过此次写入和读出的是SPI FLASH,而不是EEPROM。

30.4 下载验证

在代码编译成功之后,我们通过下载代码到ALIENTEK探索者STM32F4开发板上,通过先按KEY1按键写入数据,然后按KEY0读取数据,得到如图30.4.1所示:

 

图30.4.1 SPI实验程序运行效果图

伴随DS0的不停闪烁,提示程序在运行。程序在开机的时候会检测W25Q128是否存在,如果不存在则会在TFTLCD模块上显示错误信息,同时DS0慢闪。大家可以通过跳线帽把PB4和PB5短接就可以看到报错了。

 

 实验详细手册和源码下载地址:http://www.openedv.com/posts/list/41586.htm 

 

正点原子探索者STM32F407开发板购买地址:http://item.taobao.com/item.htm?id=41855882779

  

 

 

探索者实验连载

实验25 SPI实验.zip

 

第三十章 SPI 实验-STM32F4开发指南-正点原子探索者STM32开发板.pdf

 

http://openedv.com/thread-43360-1-1.html

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

【正点原子探索者STM32F407开发板例程连载+教学】第30章 SPI通信实验 的相关文章

  • 芯片的SD/MMC控制器以及SD卡介绍

    1 MMC SD卡 eMMC介绍 1 1 三者关联 1 最早出现的是MMC卡 卡片式结构 按照MMC协议设计 相较于NandFlash芯片来说 MMC卡有2个优势 第一是卡片化 便于拆装 第二是统一了协议接口 兼容性好 2 后来出现SD卡
  • SPI协议读写SD卡介绍

    前言 在前面的文章中 我们主要介绍了如何利用SDIO协议对SD卡进行读写操作的流程及注意事项 虽然SDIO协议读写SD卡的效率很高 但是 操作却比较麻烦 另外 还需要使用的芯片具有SDIO外设 这对于不具备SDIO外设的芯片而言 绝对是一个
  • 基于SD卡的FatFs文件系统(FatFs移植到STM32)

    平台 STM32ZET6 核心板 ST LINK V2 SD卡 USB串口线 一 移植 工程介绍 主要文件在USER组中 bsp sdio sdcard c bsp sdio sdcard h和main c 另外FatFs是用来后面移植文件
  • FATFS文件系统详解

    FATFS文件系统详解 文章目录 FATFS文件系统详解 1 简介 2 基础概念 3 FAT文件系统组成介绍 4 FAT文件系统分析 4 1 采用FAT格式格式化SD nand sd卡 4 2 引导扇区分析 4 3 分区偏移及大小计算 4
  • FATFS 0.13 f_mount(&fs,““,1)挂载失败的原因

    这两天学着用了一下FATS文件系统 虽然工作中没用到 但是对个人的经验积累还是有用的 看了一下 代码并不多 但是精简啊 指针跳来跳去的一不小心就晕了 所以也遇到了不少问题啊 这里就讲一下我遇到的第一步就懵逼的问题 那就是 FRESULT f
  • STM32的简单的SD卡读写

    导读 SD卡一般支持两种读写模式 SPI和SDIO模式 SD卡的引脚排序如下图所示 SPI模式下有几个重要的操作命令 分别是 SD卡R1回应的格式如下SPI模式下的典型初始化过程如下 1 初始化硬件配置 SPI配置 IO配置等 SD卡一般支
  • SD卡SPI模式 读写block

    声明 第一次写教程 如若有错误 请指出更正 看了很多网上的教程 还是觉得很多教程中 写多个块的时候有些问题 因此经过3天的奋斗 写出自己的教程 本教程中 没有挂载文件系统 单纯读写Block 会破坏分区和数据 下节再 装上文件系统Fatfs
  • STM32+FATFS文件系统连续在同一个txt文件里写入内容

    移植好fatfs文件系统之后 我们经常要在一个txt文件下写入数据 但是这些数据可能需要多次写入 那怎么办呢 通过这几天的学习 有以下函数可以帮助我们进行多次写入数据 1 打开文件 f open filescr2 0 201711022 t
  • SD卡/TF卡简记

    文章目录 MicroSD卡与SD卡关系与区别 对比NM卡 XQD卡 CFexpress卡 SD规格标识 FAQ 拍摄1080p或2k视频需要什么速度的sd卡 拍摄4k视频需要什么速度的sd卡 拍摄8k视频需要什么速度的sd卡 MicroSD
  • STM32 SPI方式读写SD卡

    前段时间在51上模拟SPI实现了对SD卡的读取 效果还算不错 最近将其移植到STM32上 不过使用硬件SPI和使用软件SPI还是有差别的 代码如下 void User SPIInit void GPIO InitTypeDef GPIO I
  • fatfs移植和使用(在SPI_FLASH上建立文件系统)

    文件系统对于嵌入式系统的重要性是不言而喻的 有了文件系统管理数据和外设变得方便许多 同时简化了应用的开发 今天我们来以在SPI FLASH上建立文件系统为例 看看FATFS文件系统怎么移植和使用 需要准备的材料有 1 FATFS文件系统源码
  • MMC、SD、TF、SDIO、SDMMC简介

    MMC 概念 MMC的全称是 MultiMediaCard 所以也通常被叫做 多媒体卡 是一种小巧大容量的快闪存储卡 特别应用于移动电话和数字影像及其他移动终端中 外形及接口定义 如上图所示 MMC存贮卡只有7pin 可以支持MMC和SPI
  • 发布一个在读写sd卡时,此时sd卡拔出造成死机的处理方法

    发布一个在读写sd卡时 此时sd卡拔出造成死机的处理方法 同时也在此记录一下 以备后用及有需要人们参考 stm32f4xx hal sd c 红色的代码是增加上去的 函数名称 static HAL SD ErrorTypedef SD Cm
  • STM32F407移植FATFS文件系统(版本 R0.09b)支持长文件名和中文名称

    FatFs文件系统 默认是不支持长文件名和中文名称的 要想支持长文件名和中文名称 需要打开ffconf h文件进行配置 一 支持长文件名 FatFs文件系统 默认是不支持长文件名的 要想支持长文件名 需要打开ffconf h文件进行配置 找
  • SD卡、TF卡读写速率测试

    请注意 可以在未经过博主同意下转载 但必须注明出处 谢谢 1 运行HDBENCH软件 点击确定 2 弹出如下图所示界面 配置选项并测试 选择待测TF卡 Removable可移动存储设备 图示所插入的TF卡是 I 盘 请根据实际情况选择 选择
  • SD卡 FATFS CSV 文件中的 逗号和换行

    RFC 4180 Common Format and MIME Type for Comma Separated Values CSV Files 要点有 1 CSV的换行符号要使用CRLF 即 回车符 换行符 的形式 2 文字可以使用双引
  • FATFS实现数据追加功能(原文不覆盖)

    在对FATFS的应用中我们经常需要把采集的数据存入的文件中 用作保存 也许我们的系统是一个长期的运行过程 但是我们的数据可能不是持续采集的 所以我们这样写代码 注册一个工作区域 f mount 0 fs 打开创建一个新文件 res f op
  • SD卡初始化以及命令详解

    SD卡是嵌入式设备中很常用的一种存储设备 体积小 容量大 通讯简单 电路简单所以受到很多设备厂商的欢迎 主要用来记录设备运行过程中的各种信息 以及程序的各种配置信息 很是方便 有这样几点是需要知道的 SD 卡是基于 flash 的存储卡 S
  • 关于如何修复烧写镜像文件失败的SD卡

    前言 使用某些软件 比如 win32 Disk Imager 向SD卡烧写镜像文件时 很有可能出现烧写失败的情况 通常如果烧写失败 系统会弹出请求格式化SD卡的提示框 此时不要点格式化 点了可能会造成不可挽救的结果 也可能不会 而是进行以下
  • 【正点原子探索者STM32F407开发板例程连载+教学】第30章 SPI通信实验

    第三十章 SPI 实验 mw shl code c true 1 硬件平台 正点原子探索者STM32F407开发板 2 软件平台 MDK5 1 3 固件库版本 V1 4 0 mw shl code 本章我们将向大家介绍STM32F4的SPI

随机推荐

  • Unity保存图片到相册

    Unity保存图片到Android相册 Java 纯文本查看 复制代码 01 02 03 04 05 06 07 08 09 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28
  • ESXI 7.0 版本配置N卡显卡直通

    ESXI 7 版本配置N卡显卡直通 前因 ESXI版本太新 网上啥参考资料没有 显卡直通各种问题 虚机一装显卡驱动就直接把宿主机直接整挂了 于是各种查资料 各种尝试 终于搞定直通问题 配置 名称 版本 服务器 DELL R720 ESXI
  • MySQL中的DML、DDL、DCL到底是什么玩意呢?

    个人主页 极客小俊 作者简介 web开发者 设计师 技术分享博主 希望大家多多支持一下 我们一起进步 如果文章对你有帮助的话 欢迎评论 点赞 收藏 加关注 引言 一直以来 很多人分不清这三个东西到底是什么简称 代表什么 至在面试中遇到可能会
  • nginx的location、root、alias指令用法和区别

    亲测可用 若有疑问请私信 nginx指定文件路径有两种方式root和alias 指令的使用方法和作用域 root 语法 root path 默认值 root html 配置段 http server location if alias 语法
  • Arduino动手做(48)---三轴ADXL345模块

    37款传感器与模块的提法 在网络上广泛流传 其实Arduino能够兼容的传感器模块肯定是不止37种的 鉴于本人手头积累了一些传感器和模块 依照实践 动手试试 出真知的理念 以学习和交流为目的 这里准备逐一做做实验 不管能否成功 都会记录下来
  • 第十二届蓝桥杯 2021年省赛真题 (Java 大学C组) 第二场

    蓝桥杯 2021年省赛真题 Java 大学C组 第二场 A 浮点数 B 求余 C 双阶乘 D 格点 E 整数分解 F 3 的倍数 G 特殊年份 H 小平方 I 完全平方数 J 负载均衡 A 浮点数 题目 问题描述 IEEE 754 规定一个
  • 关于HTML基本标签及结构详解

    本文主要介绍了HTML基本标签及结构详解 本文给大家介绍的非常详细 对大家的学习或工作具有一定的参考借鉴价值 需要的朋友可以参考下 1 HTML概述 1 HTML 超文本标记语言 是一种标识性语言 非编程语言 不能使用逻辑运算 通过标签将网
  • 吴恩达 deeplearning.ai课程-卷积神经网络 (2)深度卷积模型-实例探究

    参考来源 https blog csdn net red stone1 article details 78769236 https blog csdn net koala tree article details 78531398 有关C
  • python自动化笔记(九)文件操作

    文件的打开 file open test txt w encoding utf 8 参数 文件名 访问模式 write 默认为read file write hello python 删除原有内容 并写入 ret file read 读取文
  • Flutter使用百度定位经纬度数据正常,详细地址为null

    Flutter使用百度定位经纬度数据正常 详细地址为null 更新至2021 09 07 一 问题 1 使用百度定位 插件返回的数据中经纬度有正常值 其他地址信息都为null 二 分析原因 1 在wifi或移动网络没有 不好的情况下 会出现
  • 自动注册appleid

    1 通过猴油注册脚本 用js填写表单 问题 由于apple官网采用了自己封装的mvvm框架 如果只是赋值的话 还不能把视图上的数据更新到model上 必须触发一下表单元素的input事件或者change事件完成model的更新 CSP网站安
  • 孤儿进程和僵尸进程

    作者 华清远见讲师 前段时间 由于研究经典面试题 把孤儿进程和僵尸进程也总结了一下 我们有这样一个问题 孤儿进程和僵尸进程 怎么产生的 有什么危害 怎么去预防 下面是针对此问题的总结与概括 一 产生的原因 1 一般进程 正常情况下 子进程由
  • vue 表格表头内容居中

    放入
  • elasticsearch的object类型和动态映射

    我们需要讨论的最后一个自然JSON数据类型是对象 object 在其它语言中叫做hash hashmap dictionary 或者 associative array 内部对象 inner objects 经常用于在另一个对象中嵌入一个实
  • node mysql高级用法_nodejs中mysql用法

    1 建立数据库连接 createConnection Object 方法 该方法接受一个对象作为参数 该对象有四个常用的属性host user password database 与php中链接数据库的参数相同 属性列表如下 host 连接
  • Xray使用教程

    简介 Xray是长亭科技开发的一款漏扫工具 支持多种扫描方式和漏洞类型 可自定义POC Proof of Concept 概念验证 即漏洞验证程序 俺是在 乌雲安全 看到了这个工具的使用 作为一个脚本小子初学者 这里做一下笔记 使用 web
  • NVDLA系列之C-model:cvif<99>

    NV NVDLA cvif cpp WriteRequest sdp2cvif void NV NVDLA cvif WriteRequest sdp2cvif uint64 t base addr uint64 t first base
  • 通过递归,实现数组转树

    一 为什么需要数组转树 当我们做后台管理系统时难免会遇到关于公司组织架构这样的模块 一个部门下会有好几个小部门 这时我们就可以运用树形图来更好地进行查看 下面简单举例 将数组 const arr id 1 pid 0 name 生鲜 id
  • linux安装分区详解lvm,Linux下LVM的配置详解

    LVM是Logical Volume Manager 逻辑卷管理器 的简写 它为主机提供了更高层次的磁盘存储管理能力 LVM可以帮助系统管理员为应用与用户方便地分配存储空间 在LVM管理下的逻辑卷可以按需改变大小或添加移除 另外 LVM可以
  • 【正点原子探索者STM32F407开发板例程连载+教学】第30章 SPI通信实验

    第三十章 SPI 实验 mw shl code c true 1 硬件平台 正点原子探索者STM32F407开发板 2 软件平台 MDK5 1 3 固件库版本 V1 4 0 mw shl code 本章我们将向大家介绍STM32F4的SPI