基于野火指南者STM32
SPI-读写FLASH
SPI协议简介
SPI协议是由摩托罗拉公司提出的通讯协议(Serial Peripheral Interface),即串行外围设备接口,是一种高速全双工的通信总线。它被广泛地使用在ADC、LCD等设备与MCU间,要求通讯速率较高的场合。
SPI物理层的特点
•SS:从设备选择信号线,常称为片选信号线,也称为NSS、CS。
每个从设备都有独立的这一条SS信号线,本信号线独占主机的一个引脚,即有多少个从设备,就有多少条片选信号线。I2C协议中通过设备地址来寻址、选中总线上的某个设备并与其进行通讯;而SPI协议中没有设备地址,它使用SS信号线来寻址,当主机要选择从设备时,把该从设备的SS信号线设置为低电平,该从设备即被选中,即片选有效,接着主机开始与被选中的从设备进行SPI通讯。所以SPI通讯以SS线置低电平为开始信号,以SS线被拉高作为结束信号。
•SCK (Serial Clock):时钟信号线,用于通讯数据同步。
它由通讯主机产生,决定了通讯的速率,不同的设备支持的最高时钟频率不一样,如STM32的SPI时钟频率最大为fpclk/2,两个设备之间通讯时,通讯速率受限于低速设备。
•MOSI (Master Output, Slave Input):主设备输出/从设备输入引脚。
主机的数据从这条信号线输出,从机由这条信号线读入主机发送的数据,即这条线上数据的方向为主机到从机。
•MISO(Master Input,,Slave Output):主设备输入/从设备输出引脚。
主机从这条信号线读入数据,从机的数据由这条信号线输出到主机,即在这条线上数据的方向为从机到主机。
SPI的协议层
SPI协议定义了通讯的起始和停止信号、数据有效性、时钟同步等环节。
1.SPI基本通讯过程
2. 通讯的起始和停止信号
•标号1处,NSS信号线由高变低,是SPI通讯的起始信号。NSS是每个从机各自独占的信号线,当从机检在自己的NSS线检测到起始信号后,就知道自己被主机选中了,开始准备与主机通讯。
•在图中的标号6处,NSS信号由低变高,是SPI通讯的停止信号,表示本次通讯结束,从机的选中状态被取消。
3.数据有效性
•SPI使用MOSI及MISO信号线来传输数据,使用SCK信号线进行数据同步。MOSI及MISO数据线在SCK的每个时钟周期传输一位数据,且数据输入输出是同时进行的。
4.CPOL/CPHA及通讯模式
•时钟极性CPOL是指SPI通讯设备处于空闲状态时,SCK信号线的电平信号(即SPI通讯开始前、 NSS线为高电平时SCK的状态)。CPOL=0时, SCK在空闲状态时为低电平,CPOL=1时,则相反。
•时钟相位CPHA是指数据的采样的时刻,当CPHA=0时,MOSI或MISO数据线上的信号将会在SCK时钟线的“奇数边沿”被采样。当CPHA=1时,数据线在SCK的“偶数边沿”采样。
•CK信号线在空闲状态为低电平时,CPOL=0;空闲状态为高电平时,CPOL=1。
•CPHA=0,MOSI和MISO数据线的有效信号在SCK的奇数边沿保持不变,数据信号将在SCK奇数边沿时被采样,在非采样时刻,MOSI和MISO的有效信号才发生切换。
•CK信号线在空闲状态为低电平时,CPOL=0;空闲状态为高电平时,CPOL=1。
•CPHA=1,MOSI和MISO数据线的有效信号在SCK的偶数边沿保持不变,数据信号将在SCK偶数边沿时被采样,在非采样时刻,MOSI和MISO的有效信号才发生切换。
由CPOL及CPHA的不同状态,SPI分成了四种模式,主机与从机需要工作在相同的模式下才可以正常通讯,实际中采用较多的是“模式0”与“模式3”。
SPI模式 | CPOL | CPHA | 空闲时SCK时钟 | 采样时刻 |
---|---|---|---|---|
0 | 0 | 0 | 低电平 | 奇数边沿 |
1 | 0 | 1 | 低电平 | 偶数边沿 |
2 | 1 | 0 | 高电平 | 奇数边沿 |
3 | 1 | 1 | 高电平 | 偶数边沿 |
STM32的SPI外设简介
STM32的SPI外设可用作通讯的主机及从机,支持最高的SCK时钟频率为fpclk/2 (STM32F10x型号的芯片默认fpclk1为72MHz,fpclk2为36MHz),完全支持SPI协议的4种模式,数据帧长度可设置为8位或16位,可设置数据MSB先行或LSB先行。它还支持双线全双工(前面小节说明的都是这种模式)、双线单向以及单线模式。
STM32的SPI架构剖析
•通讯引脚
•时钟控制逻辑
•数据控制逻辑
•整体控制逻辑
1.通讯引脚
STM32芯片有多个SPI外设,它们的SPI通讯信号引出到不同的GPIO引脚上,使用时必须配置到这些指定的引脚,以《STM32F10x规格书》为准。
SPI1 | SPI2 | SPI3 | |
---|---|---|---|
NSS | PA4 | PB12 | PA15下载口的TDI |
CLK | PA5 | PB13 | PB3下载口的TDO |
MISO | PA6 | PB14 | PB4下载口的NTRST |
MOSI | PA7 | PB15 | PB5 |
其中SPI1是APB2上的设备,最高通信速率达36Mbtis/s,SPI2、SPI3是APB1上的设备,最高通信速率为18Mbits/s。除了通讯速率,在其它功能上没有差异。
2.时钟控制逻辑
SCK线的时钟信号,由波特率发生器根据“控制寄存器CR1”中的BR[0:2]位控制,该位是对fpclk时钟的分频因子,对fpclk的分频结果就是SCK引脚的输出时钟频率
BR[0:2] | 分频结果(SCK频率) | BR[0:2] | 分频结果(SCK频率) |
---|---|---|---|
000 | fpclk/2 | 100 | fpclk/32 |
001 | fpclk/4 | 101 | fpclk/64 |
010 | fpclk/8 | 110 | fpclk/128 |
011 | fpclk/16 | 111 | fpclk/256 |
其中的fpclk频率是指SPI所在的APB总线频率,APB1为fpclk1,APB2为fpckl2。
3.数据控制逻辑
SPI的MOSI及MISO都连接到数据移位寄存器上,数据移位寄存器的数据来源来源于接收缓冲区及发送缓冲区。
•通过写SPI的“数据寄存器DR”把数据填充到发送缓冲区中。
•通过读“数据寄存器DR”,可以获取接收缓冲区中的内容。
•其中数据帧长度可以通过“控制寄存器CR1”的“DFF位”配置成8位及16位模式;配置“LSBFIRST位”可选择MSB先行还是LSB先行。
4.整体控制逻辑
•整体控制逻辑负责协调整个SPI外设,控制逻辑的工作模式根据“控制寄存器(CR1/CR2)”的参数而改变,基本的控制参数包括前面提到的SPI模式、波特率、LSB先行、主从模式、单双向模式等等。
•在外设工作时,控制逻辑会根据外设的工作状态修改“状态寄存器(SR)”,只要读取状态寄存器相关的寄存器位,就可以了解SPI的工作状态了。除此之外,控制逻辑还根据要求,负责控制产生SPI中断信号、DMA请求及控制NSS信号线。
•实际应用中,一般不使用STM32 SPI外设的标准NSS信号线,而是更简单地使用普通的GPIO,软件控制它的电平输出,从而产生通讯起始和停止信号。
通讯过程
•控制NSS信号线,产生起始信号(图中没有画出);
•把要发送的数据写入到“数据寄存器DR”中,该数据会被存储到发送缓冲区;
•通讯开始,SCK时钟开始运行。MOSI把发送缓冲区中的数据一位一位地传输出去;MISO则把数据一位一位地存储进接收缓冲区中;
•当发送完一帧数据的时候,“状态寄存器SR”中的“TXE标志位”会被置1,表示传输完一帧,发送缓冲区已空;类似地,当接收完一帧数据的时候,“RXNE标志位”会被置1,表示传输完一帧,接收缓冲区非空;
•等待到“TXE标志位”为1时,若还要继续发送数据,则再次往“数据寄存器DR”写入数据即可;等待到“RXNE标志位”为1时,通过读取“数据寄存器DR”可以获取接收缓冲区中的内容。
假如使能了TXE或RXNE中断,TXE或RXNE置1时会产生SPI中断信号,进入同一个中断服务函数,到SPI中断服务程序后,可通过检查寄存器位来了解是哪一个事件,再分别进行处理。也可以使用DMA方式来收发“数据寄存器DR”中的数据。
就算只是接收数据也要进行发送数据动作,来产生时钟信号
SPI初始化结构体
跟其它外设一样,STM32标准库提供了SPI初始化结构体及初始化函数来配置SPI外设。初始化结构体及函数定义在库文件“stm32f10x_spi.h”及“stm32f10x_spi.c”中,编程时我们可以结合这两个文件内的注释使用或参考库帮助文档。
•SPI_Direction: 本成员设置SPI的通讯方向,可设置为双线全双工(SPI_Direction_2Lines_FullDuplex),双线只接收(SPI_Direction_2Lines_RxOnly),单线只接收(SPI_Direction_1Line_Rx)、单线只发送模式(SPI_Direction_1Line_Tx)。
•SPI_Mode: 本成员设置SPI工作在主机模式(SPI_Mode_Master)或从机模式(SPI_Mode_Slave
),这两个模式的最大区别为SPI的SCK信号线的时序,SCK的时序是由通讯中的主机产生的。若被配置为从机模式,STM32的SPI外设将接受外来的SCK信号。
•SPI_DataSize: 本成员可以选择SPI通讯的数据帧大小是为8位(SPI_DataSize_8b)还是16位(SPI_DataSize_16b)。
•SPI_CPOL和SPI_CPHA: 这两个成员配置SPI的时钟极性CPOL和时钟相位CPHA,这两个配置影响到SPI的通讯模式,时钟极性CPOL成员,可设置为高电平(SPI_CPOL_High)或低电平(SPI_CPOL_Low )。 时钟相位CPHA 则可以设置为SPI_CPHA_1Edge(在SCK的奇数边沿采集数据) 或SPI_CPHA_2Edge (在SCK的偶数边沿采集数据) 。
•SPI_NSS: 本成员配置NSS引脚的使用模式,可以选择为硬件模式(SPI_NSS_Hard )与软件模式(SPI_NSS_Soft ),在硬件模式中的SPI片选信号由SPI硬件自动产生,而软件模式则需要亲自把相应的GPIO端口拉高或置低产生非片选和片选信号。
实际中软件模式应用比较多
•SPI_BaudRatePrescaler: 本成员设置波特率分频因子,分频后的时钟即为SPI的SCK信号线的时钟频率。这个成员参数可设置为fpclk的2、4、6、8、16、32、64、128、256分频。
•SPI_FirstBit: 所有串行的通讯协议都会有MSB先行(高位数据在前)还是LSB先行(低位数据在前)的问题,而STM32的SPI模块可以通过这个结构体成员,对该特性编程控制。
•SPI_CRCPolynomial: 这是SPI的CRC校验中的多项式,若我们使用CRC校验时,就使用这个成员的参数(多项式),来计算CRC的值。
配置完这些结构体成员后,要调用SPI_Init函数把这些参数写入到寄存器中,实现SPI的初始化,然后调用SPI_Cmd来使能SPI外设。
SPI-读写串行FLASH
FLASH的存储特性:
1.在写入数据之前必须先擦除
2.擦除时会把数据为重置为1
3.写入数据时只能把为1的数据位改成0
4.擦除时必须按最小单位来擦除(在这里是扇区sector)
NOR FLASH: 可以一个字节写入
NAND FLASH:必须以块或扇区为单位进行读写
#ifndef _BSP_SPI_FLASH_H
#define _BSP_SPI_FLASH_H
#include "stm32f10x.h"
//SPI FLASH引脚定义
#define FLASH_SPI SPI1
#define FLASH_SPI_CLK RCC_APB2Periph_SPI1
#define FLASH_SPI_APBxClkCmd RCC_APB2PeriphClockCmd
#define FLASH_SPI_BAUDRATE 400000
#define FLASH_SPI_GPIO_CLK (RCC_APB2Periph_GPIOA)|(RCC_APB2Periph_GPIOC)
#define FLASH_SPI_GPIO_APBxClkCmd RCC_APB2PeriphClockCmd
#define FLASH_SPI_SCK_GPIO_PORT GPIOA
#define FLASH_SPI_SCK_GPIO_PIN GPIO_Pin_5
#define FLASH_SPI_MOSI_GPIO_PORT GPIOA
#define FLASH_SPI_MOSI_GPIO_PIN GPIO_Pin_7
#define FLASH_SPI_MISO_GPIO_PORT GPIOA
#define FLASH_SPI_MISO_GPIO_PIN GPIO_Pin_6
#define FLASH_SPI_MISO_CS_PORT GPIOC
#define FLASH_SPI_MISO_CS_PIN GPIO_Pin_0
//CS 引脚控制
#define FLASH_SPI_CS_HIGH GPIO_SetBits(FLASH_SPI_MISO_CS_PORT,FLASH_SPI_MISO_CS_PIN);
#define FLASH_SPI_CS_LOW GPIO_ResetBits(FLASH_SPI_MISO_CS_PORT,FLASH_SPI_MISO_CS_PIN);
#define DUMMY 0x00
#define READ_JEDEC_ID 0x9F
#define ERASE_SECTOR 0x20
#define READ_STATUS 0X05
#define READ_DATA 0X03
#define WRITE_ENABLE 0X06
#define WRITE_DATA 0X02
void SPI_FLASH_Init(void);
uint8_t SPI_FLASH_Send_Byte (uint8_t data);
uint32_t SPI_Read_ID(void);
void SPI_Erase_Sector(uint32_t addr);
void SPI_WaitForWriteEnd (void);
void SPI_Read_Data(uint32_t addr,uint8_t *readbuff, uint32_t numByteToRead);
void SPI_Write_Data(uint32_t addr,uint8_t *writebuff, uint32_t numByteToWrite);
#endif /*_BSP_SPI_FLASH_H*/
#include "bsp_spi_flash.h"
void Delay(uint32_t count)
{
for(;count != 0; count--);
}
void SPI_GPIO_Config(void)
{
GPIO_InitTypeDef GPIO_InitStructure;
// 打开SPI GPIO的时钟
FLASH_SPI_GPIO_APBxClkCmd(FLASH_SPI_GPIO_CLK, ENABLE);
// 打开SPI外设的时钟
FLASH_SPI_APBxClkCmd(FLASH_SPI_CLK, ENABLE);
// MOSI MISO SCK
GPIO_InitStructure.GPIO_Pin = FLASH_SPI_SCK_GPIO_PIN;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init(FLASH_SPI_SCK_GPIO_PORT, &GPIO_InitStructure);
GPIO_InitStructure.GPIO_Pin = FLASH_SPI_MOSI_GPIO_PIN;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init(FLASH_SPI_MOSI_GPIO_PORT, &GPIO_InitStructure);
GPIO_InitStructure.GPIO_Pin = FLASH_SPI_MISO_GPIO_PIN;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init(FLASH_SPI_MISO_GPIO_PORT, &GPIO_InitStructure);
//初始化CS引脚,使用软件控制,所以直接设置为推挽输出
GPIO_InitStructure.GPIO_Pin = FLASH_SPI_MISO_CS_PIN;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init(FLASH_SPI_MISO_CS_PORT, &GPIO_InitStructure);
FLASH_SPI_CS_HIGH;
}
static void SPI_Mode_Config(void)
{
SPI_InitTypeDef SPI_InitStructure;
SPI_InitStructure.SPI_BaudRatePrescaler = SPI_BaudRatePrescaler_2;
SPI_InitStructure.SPI_CPHA = SPI_CPHA_2Edge;
SPI_InitStructure.SPI_CPOL = SPI_CPOL_High;
SPI_InitStructure.SPI_CRCPolynomial = 0; //不使用,数值随便写
SPI_InitStructure.SPI_DataSize = SPI_DataSize_8b;
SPI_InitStructure.SPI_Direction = SPI_Direction_2Lines_FullDuplex;
SPI_InitStructure.SPI_FirstBit = SPI_FirstBit_MSB ;
SPI_InitStructure.SPI_Mode = SPI_Mode_Master ;
SPI_InitStructure.SPI_NSS = SPI_NSS_Soft ;
SPI_Init(FLASH_SPI,&SPI_InitStructure);
SPI_Cmd(FLASH_SPI,ENABLE);
}
void SPI_FLASH_Init()
{
SPI_GPIO_Config();
SPI_Mode_Config();
}
//发送并接收一个字节
uint8_t SPI_FLASH_Send_Byte (uint8_t data)
{
//检查并等待至TX缓冲区为空
while(SPI_I2S_GetFlagStatus(FLASH_SPI, SPI_I2S_FLAG_TXE) == RESET);
//程序执行到此处,TX缓冲区已空
SPI_I2S_SendData(FLASH_SPI,data);
//检查并等待至RX缓冲区为非空
while(SPI_I2S_GetFlagStatus(FLASH_SPI, SPI_I2S_FLAG_RXNE)== RESET);
//程序执行到此处,说明数据发送完毕,并接受到一个字节
return SPI_I2S_ReceiveData(FLASH_SPI);
}
uint8_t SPI_FLASH_Read_Byte(void)
{
return SPI_FLASH_Send_Byte (DUMMY);
}
uint32_t SPI_Read_ID(void)
{
uint32_t flash_id;
FLASH_SPI_CS_LOW;
SPI_FLASH_Send_Byte (READ_JEDEC_ID);
flash_id = SPI_FLASH_Send_Byte (DUMMY);
flash_id <<= 8;
flash_id |= SPI_FLASH_Send_Byte (DUMMY);
flash_id <<= 8;
flash_id |= SPI_FLASH_Send_Byte (DUMMY);
FLASH_SPI_CS_HIGH
return flash_id;
}
void SPI_Write_Enable (void)
{
FLASH_SPI_CS_LOW;
SPI_FLASH_Send_Byte (WRITE_ENABLE);
FLASH_SPI_CS_HIGH;
}
void SPI_Erase_Sector(uint32_t addr)
{
SPI_Write_Enable ();
FLASH_SPI_CS_LOW;
SPI_FLASH_Send_Byte (ERASE_SECTOR);
SPI_FLASH_Send_Byte ((addr >> 16) & 0xff);
SPI_FLASH_Send_Byte ((addr >> 8) & 0xff);
SPI_FLASH_Send_Byte (addr & 0xff);
FLASH_SPI_CS_HIGH
SPI_WaitForWriteEnd(); //等待内部擦除时序完成
}
void SPI_WaitForWriteEnd (void)
{
uint8_t status_reg = 0;
SPI_Write_Enable ();
FLASH_SPI_CS_LOW;
SPI_FLASH_Send_Byte (READ_STATUS);
do
{
status_reg = SPI_FLASH_Send_Byte (DUMMY);
}while ((status_reg & 0x01) == 1); //忙碌
FLASH_SPI_CS_HIGH
}
void SPI_Read_Data(uint32_t addr,uint8_t *readbuff, uint32_t numByteToRead)
{
FLASH_SPI_CS_LOW;
SPI_FLASH_Send_Byte (READ_DATA);
SPI_FLASH_Send_Byte ((addr >> 16) & 0xff);
SPI_FLASH_Send_Byte ((addr >> 8) & 0xff);
SPI_FLASH_Send_Byte (addr & 0xff);
while(numByteToRead--)
{
*readbuff = SPI_FLASH_Send_Byte(DUMMY);
readbuff++;
}
FLASH_SPI_CS_HIGH;
SPI_WaitForWriteEnd ();
}
void SPI_Write_Data(uint32_t addr,uint8_t *writebuff, uint32_t numByteToWrite)
{
SPI_Write_Enable ();
FLASH_SPI_CS_LOW;
SPI_FLASH_Send_Byte (WRITE_DATA);
SPI_FLASH_Send_Byte ((addr >> 16) & 0xff);
SPI_FLASH_Send_Byte ((addr >> 8) & 0xff);
SPI_FLASH_Send_Byte (addr & 0xff);
while(numByteToWrite--)
{
SPI_FLASH_Send_Byte(*writebuff);
writebuff++;
}
FLASH_SPI_CS_HIGH;
SPI_WaitForWriteEnd ();
}
#include "stm32f10x.h"
#include "bsp_led.h"
#include "bsp_usart.h"
#include "./FLASH/bsp_spi_flash.h"
uint8_t readbuff[4096];
uint8_t writebuff[4096];
int main(void)
{
uint32_t id;
uint16_t i;
USART_Config();
SPI_FLASH_Init();
printf("这是一个SPI读写串行FLASH通讯实验\n");
id = SPI_Read_ID();
printf("\r\n id = 0x%x\r\n",id);
SPI_Erase_Sector(0);
for(i = 0;i<25;i++)
{
writebuff[i] = i;
}
SPI_Write_Data(0,writebuff, 25);
SPI_Read_Data(0,readbuff,4096);
for(i = 0;i<4096;i++)
{
printf("0x%x ",readbuff[i]);
if(i%10 == 0)
printf("\r\n");
}
while(1)
{}
}