需求描述
基于寄存器操作,使用软件模拟SPI协议,完成读写FLASH。
状态寄存器

硬件电路设计


寄存器实现
spi.h
#ifndef __SPI_H
#define __SPI_H
#include "stm32f10x.h"
#include "delay.h"
// 宏定义,不同引脚输出高低电平
#define CS_HIGH (GPIOC->ODR |= GPIO_ODR_ODR13)
#define CS_LOW (GPIOC->ODR &= ~GPIO_ODR_ODR13)
#define SCK_HIGH (GPIOA->ODR |= GPIO_ODR_ODR5)
#define SCK_LOW (GPIOA->ODR &= ~GPIO_ODR_ODR5)
#define MOSI_HIGH (GPIOA->ODR |= GPIO_ODR_ODR7)
#define MOSI_LOW (GPIOA->ODR &= ~GPIO_ODR_ODR7)
// 读取MISO引脚信号
#define MISO_READ (GPIOA->IDR & GPIO_IDR_IDR6)
// SPI标准延迟时间
#define SPI_DELAY Delay_us(5)
// 初始化
void SPI_Init(void);
// 数据传输的开始和结束
void SPI_Start(void);
void SPI_Stop(void);
// 主从设备交换一个字节的数据
uint8_t SPI_SwapByte(uint8_t byte);
#endif
spi.c
#include "spi.h"
// 初始化
void SPI_Init(void)
{
// 1. 开启时钟
RCC->APB2ENR |= RCC_APB2ENR_IOPAEN;
RCC->APB2ENR |= RCC_APB2ENR_IOPCEN;
// 2. GPIO工作模式
// PA5、PA7、PC13:通用推挽输出,CNF = 00,MODE = 11
GPIOC->CRH |= GPIO_CRH_MODE13;
GPIOC->CRH &= ~GPIO_CRH_CNF13;
GPIOA->CRL |= GPIO_CRL_MODE5;
GPIOA->CRL &= ~GPIO_CRL_CNF5;
GPIOA->CRL |= GPIO_CRL_MODE7;
GPIOA->CRL &= ~GPIO_CRL_CNF7;
// PA6:MISO,浮空输入,CNF = 01,MODE = 00
GPIOA->CRL &= ~GPIO_CRL_MODE6;
GPIOA->CRL &= ~GPIO_CRL_CNF6_1;
GPIOA->CRL |= GPIO_CRL_CNF6_0;
// 3. 选择SPI的工作模式 0:SCK空闲0
SCK_LOW;
// 4. 片选不使能
CS_HIGH;
// 5. 延时
SPI_DELAY;
}
// 数据传输的开始和结束
void SPI_Start(void)
{
CS_LOW;
}
void SPI_Stop(void)
{
CS_HIGH;
}
// 主从设备交换一个字节的数据
uint8_t SPI_SwapByte(uint8_t byte)
{
// 定义变量保存接收到的字节
uint8_t rByte = 0x00;
// 用一个循环,依次交换8位数据
for (uint8_t i = 0; i < 8; i++)
{
// 1. 先准备要发送的数据(最高位),送到MOSI
if (byte & 0x80)
{
MOSI_HIGH;
}
else
{
MOSI_LOW;
}
// 左移一位
byte <<= 1;
// 2. 拉高时钟信号,形成一个上升沿
SCK_HIGH;
SPI_DELAY;
// 3. 在MISO上采样Flash传来的数据
// 先做左移
rByte <<= 1;
if (MISO_READ)
{
rByte |= 0x01;
}
// 4. 拉低时钟,为下次数据传输做准备
SCK_LOW;
SPI_DELAY;
}
return rByte;
}
w25q32.h
#ifndef __W25Q32_H
#define __W25Q32_H
#include "spi.h"
// 初始化
void W25Q32_Init(void);
// 读取ID
void W25Q32_ReadID(uint8_t * mid, uint16_t * did);
// 开启写使能
void W25Q32_WriteEnable(void);
// 关闭写使能
void W25Q32_WriteDisable(void);
// 等待状态不为忙(busy)
void W25Q32_WaitNotBusy(void);
// 擦除段(sector erase),地址只需要块号和段号
void W25Q32_EraseSector(uint8_t block, uint8_t sector);
// 写入(页写)
void W25Q32_PageWrite(uint8_t block, uint8_t sector, uint8_t page, uint8_t * data, uint16_t len);
// 读取
void W25Q32_Read(uint8_t block, uint8_t sector, uint8_t page, uint8_t innerAddr, uint8_t * buffer, uint16_t len);
#endif
w25q32.c
#include "w25q32.h"
// 初始化
void W25Q32_Init(void)
{
SPI_Init();
}
// 读取ID
void W25Q32_ReadID(uint8_t *mid, uint16_t *did)
{
SPI_Start();
// 1. 发送指令 9fh
SPI_SwapByte(0x9f);
// 2. 获取制造商ID(为了读取数据,发送什么不重要)
*mid = SPI_SwapByte(0xff);
// 3. 获取设备ID
*did = 0;
*did |= SPI_SwapByte(0xff) << 8;
*did |= SPI_SwapByte(0xff) & 0xff;
SPI_Stop();
}
// 开启写使能
void W25Q32_WriteEnable(void)
{
SPI_Start();
SPI_SwapByte(0x06);
SPI_Stop();
}
// 关闭写使能
void W25Q32_WriteDisable(void)
{
SPI_Start();
SPI_SwapByte(0x04);
SPI_Stop();
}
// 等待状态不为忙(busy)
void W25Q32_WaitNotBusy(void)
{
SPI_Start();
// 发送读取状态寄存器指令,发送指令之后,FLASH会一直返回状态寄存器的值。其中最后一位是忙状态的标志位。
SPI_SwapByte(0x05);
// 等待收到的数据末位变成0
while (SPI_SwapByte(0xff) & 0x01)
{
}
SPI_Stop();
}
// 擦除段(sector erase),地址只需要块号和段号
void W25Q32_EraseSector(uint8_t block, uint8_t sector)
{
// 首先等待状态不为忙
W25Q32_WaitNotBusy();
// 开启写使能
W25Q32_WriteEnable();
// 计算要发送的地址(段首地址)
uint32_t addr = (block << 16) + (sector << 12);
SPI_Start();
// 发送指令
SPI_SwapByte(0x20);
SPI_SwapByte(addr >> 16 & 0xff); // 第一个字节
SPI_SwapByte(addr >> 8 & 0xff); // 第二个字节
SPI_SwapByte(addr >> 0 & 0xff); // 第三个字节
SPI_Stop();
W25Q32_WriteDisable();
}
// 写入(页写)
void W25Q32_PageWrite(uint8_t block, uint8_t sector, uint8_t page, uint8_t *data, uint16_t len)
{
// 首先等待状态不为忙
W25Q32_WaitNotBusy();
// 开启写使能
W25Q32_WriteEnable();
// 计算要发送的地址(页首地址)
uint32_t addr = (block << 16) + (sector << 12) + (page << 8);
SPI_Start();
// 发送指令
SPI_SwapByte(0x02);
// 发送24位地址
SPI_SwapByte(addr >> 16 & 0xff); // 第一个字节
SPI_SwapByte(addr >> 8 & 0xff); // 第二个字节
SPI_SwapByte(addr >> 0 & 0xff); // 第三个字节
// 依次发送数据
for (uint16_t i = 0; i < len; i++)
{
SPI_SwapByte(data[i]);
}
SPI_Stop();
W25Q32_WriteDisable();
}
// 读取
void W25Q32_Read(uint8_t block, uint8_t sector, uint8_t page, uint8_t innerAddr, uint8_t *buffer, uint16_t len)
{
// 首先等待状态不为忙
W25Q32_WaitNotBusy();
// 计算要发送的地址
uint32_t addr = (block << 16) + (sector << 12) + (page << 8) + innerAddr;
SPI_Start();
// 发送指令
SPI_SwapByte(0x03);
// 发送24位地址
SPI_SwapByte(addr >> 16 & 0xff); // 第一个字节
SPI_SwapByte(addr >> 8 & 0xff); // 第二个字节
SPI_SwapByte(addr >> 0 & 0xff); // 第三个字节
// 依次读取数据
for (uint16_t i = 0; i < len; i++)
{
buffer[i] = SPI_SwapByte(0xff);
}
SPI_Stop();
}
main.c
#include "usart.h"
#include "w25q32.h"
#include <string.h>
int main(void)
{
// 1. 初始化
USART_Init();
W25Q32_Init();
printf("SPI软件模拟实验开始...\n");
// 2. 读取ID进行测试
uint8_t mid = 0;
uint16_t did = 0;
W25Q32_ReadID(&mid, &did);
printf("mid = %#x, did = %#x\n", mid, did);
// 3. 段擦除
W25Q32_EraseSector(0, 0);
// 4. 页写
W25Q32_PageWrite(0, 0, 0, "12345678", 8);
// 5. 读取
uint8_t buffer[10] = {0};
W25Q32_Read(0, 0, 0, 2, buffer, 6);
printf("buffer = %s\n", buffer);
while (1)
{
}
}