STM32 SPI(二)软件读写W25Q64
程序整体框架
- 先建一个MySPI的模块,在这个模块里,主要包含通信引脚封装、初始化,以及SPI通信的3个时序基本单元,起始、终止和交换一个字节,这是SPI通信层的内容。
- 基于SPI通信层,再建一个W25Q64的模块,在这个模块里,调用底层SPI的拼图,来拼接各种指令和功能的完整时序,比如写使能、擦除、页编程、读数据等等,这是W25Q64硬件驱动层的内容。
- 最后在主函数里,调用驱动层的函数,来完成我们想要实现的功能。
程序示例
MySPI.c
#include "stm32f10x.h" // Device header
void MySPI_W_SS(uint8_t BitValue)//从机选择
{
GPIO_WriteBit(GPIOA,GPIO_Pin_4,(BitAction)BitValue);//将BitValue强转为BitAction枚举类型
}
void MySPI_W_SCK(uint8_t BitValue)//时钟
{
GPIO_WriteBit(GPIOA,GPIO_Pin_5,(BitAction)BitValue);//将BitValue强转为BitAction枚举类型
}
void MySPI_W_MOSI(uint8_t BitValue)//主机输出从机输入
{
GPIO_WriteBit(GPIOA,GPIO_Pin_7,(BitAction)BitValue);//将BitValue强转为BitAction枚举类型
}
uint8_t MySPI_R_MISO(void)//主机输入从机输出
{
return GPIO_ReadInputDataBit(GPIOA,GPIO_Pin_6);
}
void MySPI_Init(void)
{
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);//打开GPIOA的时钟
GPIO_InitTypeDef GPIO_InitStructure;
GPIO_InitStructure.GPIO_Mode=GPIO_Mode_Out_PP;//推挽输出
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_4|GPIO_Pin_5|GPIO_Pin_7;//将主机的输出引脚(时钟、主机输出和片选)配置为推挽输出 SS-PA4 SCK-PA5 MOSI-PA7
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init(GPIOA,&GPIO_InitStructure);
GPIO_InitStructure.GPIO_Mode=GPIO_Mode_IPU;//上拉输入
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6;//将主机的输入(主机输入)引脚配置为上拉输入 MISO-PA6
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init(GPIOA,&GPIO_InitStructure);//初始化GPIOA
//配置初始化之后引脚的默认电平
MySPI_W_SS(1);//SS置高电平,默认不选择从机
MySPI_W_SCK(0);//使用SPI模式0,所以默认是低电平
}
void MySPI_Start(void)//起始条件
{
MySPI_W_SS(0);//SS从高电平转换到低电平
}
void MySPI_Stop(void)//终止条件
{
MySPI_W_SS(1);//SS从低电平转换到高电平
}
uint8_t MySPI_SwapByte(uint8_t ByteSend)//交换一个字节 ByteSend是我们传进来的参数,要通过交换一个字节的时序发送出去,返回值是ByteReceive,是通过交换一个字节接收到的数据。
{
uint8_t i,ByteReceive=0x00;
for(i=0;i<8;i++)
{
MySPI_W_MOSI(ByteSend & (0x80>>i));//通过掩码,依次提取数据每一位进行操作
//主机依次移出对应位的数据到MOSI上 0x80>>i的作用就是用来挑出数据的某一位或者某几位,或者换种描述方式就是,用来屏 蔽其他的无关,我们可以把这种类型的数据叫做掩码
MySPI_W_SCK(1);//SCK产生上升沿时,从机会自动把MOSI的数据读走
if(MySPI_R_MISO()==1)
{
ByteReceive |= (0x80>>i);//主机依次读取从机放到MISO上的数据
}
MySPI_W_SCK(0);//SCK产生下降沿,主机和从机移出下一位
}
return ByteReceive;//主机和从机置换数据
}
//uint8_t MySPI_SwapByte(uint8_t ByteSend)//移位模型
//{
// uint8_t i;
// for(i=0;i<8;i++)
// {
// MySPI_W_MOSI(ByteSend & 0x80);
// ByteSend <<=1;//左移移位,最低位会自动补0,最低位空缺
// MySPI_W_SCK(1);
// if(MySPI_R_MISO()==1)
// {
// ByteSend |= 0x01;//把收到的数据放在ByteSend的最低位
// }
// MySPI_W_SCK(0);
// }
// return ByteSend;
//}
MySPI.h
#ifndef __MYSPI_H
#define __MYSPI_H
void MySPI_Init(void);//SPI初始化
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" //指令码宏定义,增加可读性
void W25Q64_Init(void)
{
MySPI_Init();
}
void W25Q64_ReadID(uint8_t *MID,uint16_t *DID)//读取ID号 用指针实现多返回值
{
MySPI_Start();//开始条件
MySPI_SwapByte(W25Q64_JEDEC_ID);//发送读取ID号的指令码
*MID = MySPI_SwapByte(W25Q64_DUMMY_BYTE);//用无用数据置换回厂商ID
*DID = MySPI_SwapByte(W25Q64_DUMMY_BYTE);//用无用数据置换回设备ID 设备ID高8位,表示存储器类型;低8位,表示容量
*DID <<= 8;//把第一次读到的设备ID高八位移到DID的高8位去
*DID |= MySPI_SwapByte(W25Q64_DUMMY_BYTE);//将高8位和低8位的数据拼接起来,得到完整16位设备ID
MySPI_Stop();//终止条件
}
void W25Q64_WriteEnable(void)//写使能
{
MySPI_Start();//开始条件
MySPI_SwapByte(W25Q64_WRITE_ENABLE);//发送写使能的指令码
MySPI_Stop();//终止条件
}
void W25Q64_WaitBusy(void)//等待BUSY为0(芯片结束忙状态)读取状态寄存器1,判断芯片是否处于忙状态,状态寄存器可以连续读出,方便我们等待
{
uint32_t Timeout;//防止超时
MySPI_Start();//开始条件
MySPI_SwapByte(W25Q64_READ_STATUS_REGISTER_1);//发送读取状态寄存器1的指令码
Timeout = 100000;
while((MySPI_SwapByte(W25Q64_DUMMY_BYTE) & 0x01)==0x01)//用掩码取出最低位,读取状态寄存器1的最低位BUSY,判断其是否为1,1表示芯片处于忙状态,0表示芯片结束忙状态。
{
Timeout --;
if (Timeout==0)
{
break;//超时等待,跳出循环
}
}
MySPI_Stop();//终止条件
}
void W25Q64_PageProgram(uint32_t Address,uint8_t *DataArray,uint16_t Count)//页编程 通过指针传递数组
{
uint16_t i;
W25Q64_WriteEnable();//写入操作前,必须先进行写使能,写使能仅对之后跟随的一条时序有效,一条时序结束后,写失能。
MySPI_Start();//开始条件
MySPI_SwapByte(W25Q64_PAGE_PROGRAM);//发送页编程的指令码 地址0x12 34 56
MySPI_SwapByte(Address>>16);//Address右移16位,发送最高字节的地址 0x12
MySPI_SwapByte(Address>>8);//Address0右移8位,发送次高位字节的地址,由于交换字节函数只能接收8位数据,所以高位舍弃,实际发送0x34,就是中间的字节 0x34
MySPI_SwapByte(Address);//舍弃高位,就是最低位的字节 0x56
for(i=0;i<Count;i++)
{
MySPI_SwapByte(DataArray[i]);//发送写入的数据
}
MySPI_Stop();//终止条件
W25Q64_WaitBusy();//事后等待,等芯片结束忙状态再退出。因为写入操作结束后,芯片进入忙状态,不响应新的读写操作。
}
void W25Q64_SectorErace(uint32_t Address)//扇区擦除 擦除指定地址所在的整个扇区
{
W25Q64_WriteEnable();//写入操作前,必须先进行写使能,写使能仅对之后跟随的一条时序有效,一条时序结束后,写失能。保证函数结束后,芯片不处于忙状态。
MySPI_Start();//开始条件
MySPI_SwapByte(W25Q64_SECTOR_ERASE_4KB);//发送扇区擦除的指令码
MySPI_SwapByte(Address>>16);
MySPI_SwapByte(Address>>8);
MySPI_SwapByte(Address);
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);
MySPI_SwapByte(Address>>8);
MySPI_SwapByte(Address);
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
#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
#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
main.c
#include "stm32f10x.h" // Device header
#include "OLED.h"
#include "W25Q64.h"
uint8_t MID;
uint16_t DID;
uint8_t ArrayWrite[]={0x01,0x02,0x03,0x04};
uint8_t ArrayRead[4];
int main(void)
{
OLED_Init();
W25Q64_Init();
OLED_ShowString(1,1,"MID: DID:");
OLED_ShowString(2,1,"W:");
OLED_ShowString(3,1,"R:");
W25Q64_ReadID(&MID,&DID);
OLED_ShowHexNum(1,5,MID,2);
OLED_ShowHexNum(1,12,DID,4);
W25Q64_SectorErace(0x000000);
W25Q64_PageProgram(0x000000,ArrayWrite,4);
W25Q64_ReadData(0x000000,ArrayRead,4);
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);
while(1)
{
}
}
注意事项
- Flash 数据掉电不丢失
- Flash 擦除之后数据变为FF
- Flash 只能1写0,不能0写1,所以写入数据前必须进行擦除
- 写入数据不能跨页,页地址的范围是xxxx00到xxxxFF,所以最后两个16进制数是页内地址,前面四位是页地址。如果从一页的最后一个地址开始写,写入的数据不能跨页,后面的数据会返回页首。而读取可以跨页,所以可以读取到第二页的数据。
W25Q64_WaitBusy()事前等待还是事后等待
- 事后等待,在每次写入后,等待状态寄存器1的BUSY位清零,再退出,这样函数结束后,芯片肯定结束忙状态。
- 事前等待,在每次操作之前,进行等待。在每次写入之前,等待状态寄存器1的BUSY位清零。在芯片不处于忙状态的时候,再写入。
- 事后等待,最保险,在函数以外的地方,芯片肯定不处于忙状态;事前等待,效率会高一些,因为写完之后不需要等待,程序可以执行其他代码,可以正好利用执行其他代码的时间,来消耗等待的时间,可能下一次事前等待的时候,时间被执行其他代码耗过去了,就不需要等待了。
- 事后等待,只需要在写入之后调用;而事前等待,在写入操作和读取操作之前,都得调用,因为在芯片处于忙状态时,不响应新的读写操作。