一.硬件SPI基本信息
STM32内部集成了硬件SPI收发电路,可以由硬件自动执行时序生成、数据收发等共功能
可配置8/16位数据帧、高位先行/低位先行
时钟频率:fpclk/(2,4,8,16,32,64,128,256)
可精简为半双工/单工通信
支持DMA/兼容I2S(数字音频传输协议
二.硬件SPI基本结构
SPE:配置SPE使能
BR:配置波特率也就是SCK时钟频率
MSTR:配置主从模式:1是主模式,0是从模式
SPI接口:LSBFIRST(帧格式)可以控制高位先行或者低位先行,MSB(0)高位先行,LSB(1)低位先行
1.SPI基本结构
开启时钟
初始化GPIO:SCK与MOSI复用推挽输出(AF_PP),MISO上拉输入(IPU),SS通用推挽输出(OUT_PP)
配置SPI外设:选择主模式(Master),双线全双工(2Lines_FullDuplex),8位数据帧(DataSize_8b)高位先行(FirstBit_MSB),波特率128(BaudRatePrescaler_128),NSS软件(Soft),CRC校验多项式给默认值7(CRCPolynomial)
SPI使能
2.SPI基本时序
发送缓存区(寄存器):TDR TXE发送寄存器置1表示空,以写入数据开始传输
接收缓存区(寄存器):RDR RXNE接受寄存器置1表示非空,接收到了数据
连续传输:追求极限频率
非连续传输:字节间隔会随着频率趋于高速而显现
时序流程:等待标志位SPI置1,之后发送数据到TDR,TDR自动转入移位寄存器,然后数据会通过MOSI一位一位移出,就会在MOSI线上产生时序,等待RXNE置1表示收到了数据,时序完成,返回接收寄存器(RDR)的值
三.代码实现
main.c
#include "stm32f10x.h" // Device header
#include "Delay.h"
#include "OLED.h"
#include "W25Q64.h"
uint8_t ArrayWrite[] = {0x01,0x02,0x03,0x04};//初始化写入的变量
uint8_t ArrayRead[4];//初始化读取的变量
uint8_t MID;
uint16_t DID;
int main(void)
{
OLED_Init ();
W25Q64_Init();
W25Q64_ReadID(&MID,&DID);
OLED_ShowString (1,1,"MID:");
OLED_ShowString (1,8,"DID:");
OLED_ShowString(2,1,"W:");
OLED_ShowString(3,1,"R:");
OLED_ShowHexNum (1,5,MID,2);
OLED_ShowHexNum (1,12,DID,4);
W25Q64_SectorErase(0x000000);
W25Q64_PageProgram(0x000000,ArrayWrite,4);
W25Q64_ReadData(0x000000,ArrayRead,4);
while(1)
{
OLED_ShowHexNum(2,3,ArrayWrite[0],2);
OLED_ShowHexNum(2,6,ArrayWrite[1],2);
OLED_ShowHexNum(2,9,ArrayWrite[2],2);
OLED_ShowHexNum(2,12,ArrayWrite[3],2);
OLED_ShowHexNum(3,3,ArrayRead[0],2);
OLED_ShowHexNum(3,6,ArrayRead[1],2);
OLED_ShowHexNum(3,9,ArrayRead[2],2);
OLED_ShowHexNum(3,12,ArrayRead[3],2);
}
}
MySPI.c
#include "stm32f10x.h" // Device header
//初始化SS
void MySPI_W_SS(uint8_t BitValue)
{
GPIO_WriteBit(GPIOA,GPIO_Pin_4,(BitAction)BitValue );//SS使用PA4
}
//封装MySPI函数
void MySPI_Init(void)
{
//开启时钟
RCC_APB2PeriphClockCmd (RCC_APB2Periph_GPIOA,ENABLE );//开启GPIOA时钟
RCC_APB2PeriphClockCmd (RCC_APB2Periph_SPI1 ,ENABLE);//开启SPI1时钟
//初始化GPIO
GPIO_InitTypeDef GPIO_InitStructure;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;//输出口通用推挽输出模式
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_4;//CS(SS)从机输出PA4
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;//速度50
GPIO_Init(GPIOA,&GPIO_InitStructure);
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;//输出口复用推挽输出模式
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_5 | GPIO_Pin_7;//SCK(SCL)时钟输出,DI(MOSI)主机输出从机输入
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;//速度50
GPIO_Init(GPIOA,&GPIO_InitStructure);
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU;//输入口上拉输入
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6;//DO(MISO)主机输入从机输出
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;//速度50
GPIO_Init(GPIOA,&GPIO_InitStructure);
//初始化SPI
SPI_InitTypeDef SPI_InitStructure;
SPI_InitStructure.SPI_BaudRatePrescaler = SPI_BaudRatePrescaler_128;//波特率128
SPI_InitStructure.SPI_CPHA = SPI_CPHA_1Edge;//0模式,CPHA=0,下降沿发送
SPI_InitStructure.SPI_CPOL = SPI_CPOL_Low;//0模式,CPOL= 0,低电平空闲
SPI_InitStructure.SPI_CRCPolynomial = 7;//CRC校验多选项给默认值7
SPI_InitStructure.SPI_DataSize = SPI_DataSize_8b;//选择8位数据大小
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;//NSS软件
SPI_Init(SPI1,&SPI_InitStructure);
//使能SPI
SPI_Cmd(SPI1,ENABLE);
//默认电平
MySPI_W_SS(1);//从机默认高电平
}
//开始条件
void MySPI_Start(void)
{
MySPI_W_SS(0);//从机低电平开始
}
//终止条件
void MySPI_Stop(void)
{
MySPI_W_SS(1);//从机高电平
}
//交换数据
uint8_t MySPI_SwapByte(uint8_t ByteSend)
{
while(SPI_I2S_GetFlagStatus(SPI1,SPI_I2S_FLAG_TXE) != SET);//判断标志位TXE是否置1(可写入数据)
SPI_I2S_SendData(SPI1,ByteSend);//MOSI线发送数据
while(SPI_I2S_GetFlagStatus(SPI1,SPI_I2S_FLAG_RXNE) != SET);//判断标志位RXNE是否置1(已经接到数据)
return SPI_I2S_ReceiveData(SPI1);//MISO接收数据
}
MySPI.h
#ifndef _MYSPI_H
#define _MYSPI_H
void MySPI_Init(void);
void MySPI_Start(void);
void MySPI_Stop(void);
uint8_t MySPI_SwapByte(uint8_t ByteSend);
#endif
W25Q64.c
#include "stm32f10x.h" // Device header
#include "MySPI.h"
#include "W25Q64_Ins.h"
//W25Q64函数封装
void W25Q64_Init(void)
{
MySPI_Init();
}
//获取ID数据函数
void W25Q64_ReadID(uint8_t *MID,uint16_t *DID)
{
MySPI_Start ();//开始条件
MySPI_SwapByte(W25Q64_JEDEC_ID);//指令码
*MID = MySPI_SwapByte(W25Q64_DUMMY_BYTE);//交换数据
*DID = MySPI_SwapByte(W25Q64_DUMMY_BYTE);//交换数据
*DID <<= 8;//高8位右移
*DID |= MySPI_SwapByte(W25Q64_DUMMY_BYTE);//交换高8位+低8位指定地址数据
MySPI_Stop ();//终止条件
}
//使能函数
void W25Q64_WriteEnable(void)
{
MySPI_Start ();//开始
MySPI_SwapByte(W25Q64_WRITE_ENABLE);//指定使能地址
MySPI_Stop ();//结束
}
//等待忙函数
void W25Q64_WaitBusy(void)
{
uint32_t Timeout;
MySPI_Start ();//开始
MySPI_SwapByte(W25Q64_READ_STATUS_REGISTER_1);//读状态寄存器1地址
Timeout = 100000;
while((MySPI_SwapByte(W25Q64_DUMMY_BYTE) & 0x01) == 0x01)//等待垃圾地址数据读完
{
Timeout --;//自减
if(Timeout == 0)//自减到0结束程序
{
break;
}
}
MySPI_Stop ();//结束
}
//擦除函数
void W25Q64_SectorErase(uint32_t Address)
{
W25Q64_WriteEnable();
MySPI_Start ();//开始
MySPI_SwapByte(W25Q64_SECTOR_ERASE_4KB);//指定擦除指令
MySPI_SwapByte(Address >> 16);//前8位指定地址
MySPI_SwapByte(Address >> 8);//中8位指定地址
MySPI_SwapByte(Address);//后8位指定地址
MySPI_Stop ();//结束
W25Q64_WaitBusy();
}
//页编程函数
void W25Q64_PageProgram(uint32_t Address,uint8_t *DataArray,uint16_t Count)//指定地址,指定数据,指定次数
{
uint32_t i;
W25Q64_WriteEnable();
MySPI_Start ();//开始
MySPI_SwapByte(W25Q64_PAGE_PROGRAM);//指定页编程指令
MySPI_SwapByte(Address >> 16);//前8位指定地址
MySPI_SwapByte(Address >> 8);//中8位指定地址
MySPI_SwapByte(Address);//后8位指定地址
for(i = 0;i < Count;i++)
{
MySPI_SwapByte(DataArray[i]);//写入指定次数指定数据
}
MySPI_Stop ();//结束
W25Q64_WaitBusy();
}
//指定读函数
void W25Q64_ReadData(uint32_t Address,uint8_t *DataArray,uint32_t Count)//指定地址,指定数据,指定次数
{
uint32_t i;
MySPI_Start ();//开始
MySPI_SwapByte(W25Q64_READ_DATA);//指定读数据指令
MySPI_SwapByte(Address >> 16);//前8位指定地址
MySPI_SwapByte(Address >> 8);//中8位指定地址
MySPI_SwapByte(Address);//后8位指定地址
for(i = 0;i < Count;i++)
{
DataArray[i] = MySPI_SwapByte(W25Q64_DUMMY_BYTE);//写入指定次数指定数据
}
MySPI_Stop ();//结束
}
W25Q64_Ins.h
#ifndef __W25Q64_INS_H
#define __W25Q64_INS_H
#define W25Q64_WRITE_ENABLE 0x06//写指令使能指令
#define W25Q64_WRITE_DISABLE 0x04
#define W25Q64_READ_STATUS_REGISTER_1 0x05//读状态寄存器1指令
#define W25Q64_READ_STATUS_REGISTER_2 0x35
#define W25Q64_WRITE_STATUS_REGISTER 0x01
#define W25Q64_PAGE_PROGRAM 0x02
#define W25Q64_QUAD_PAGE_PROGRAM 0x32//读读页编程地址
#define W25Q64_BLOCK_ERASE_64KB 0xD8
#define W25Q64_BLOCK_ERASE_32KB 0x52
#define W25Q64_SECTOR_ERASE_4KB 0x20//擦除扇形指令
#define W25Q64_CHIP_ERASE 0xC7
#define W25Q64_ERASE_SUSPEND 0x75
#define W25Q64_ERASE_RESUME 0x7A
#define W25Q64_POWER_DOWN 0xB9
#define W25Q64_HIGH_PERFORMANCE_MODE 0xA3
#define W25Q64_CONTINUOUS_READ_MODE_RESET 0xFF
#define W25Q64_RELEASE_POWER_DOWN_HPM_DEVICE_ID 0xAB
#define W25Q64_MANUFACTURER_DEVICE_ID 0x90
#define W25Q64_READ_UNIQUE_ID 0x4B
#define W25Q64_JEDEC_ID 0x9F//读ID指令
#define W25Q64_READ_DATA 0x03//读数据指令
#define W25Q64_FAST_READ 0x0B
#define W25Q64_FAST_READ_DUAL_OUTPUT 0x3B
#define W25Q64_FAST_READ_DUAL_IO 0xBB
#define W25Q64_FAST_READ_QUAD_OUTPUT 0x6B
#define W25Q64_FAST_READ_QUAD_IO 0xEB
#define W25Q64_OCTAL_WORD_READ_QUAD_IO 0xE3
#define W25Q64_DUMMY_BYTE 0xFF//垃圾地址,用来换数据指令
#endif
W25Q64.h
#ifndef __W25Q64_H
#define __W25Q64_H
void W25Q64_Init(void);
void W25Q64_ReadID(uint8_t *MID, uint16_t *DID);
void W25Q64_PageProgram(uint32_t Address, uint8_t *DataArray, uint16_t Count);
void W25Q64_SectorErase(uint32_t Address);
void W25Q64_ReadData(uint32_t Address, uint8_t *DataArray, uint32_t Count);
#endif