STM32微控制器的SPI通信:硬件与软件实现W25Q64 Flash读写

摘要

本文将详细介绍STM32微控制器如何通过硬件SPI和模拟SPI两种方式来实现与W25Q64 Flash存储器的通信。W25Q64是一款8Mbit的串行Flash存储器,广泛应用于嵌入式系统中。本文将包括SPI通信原理、W25Q64特性、硬件连接、SPI初始化、读写操作及示例代码。

1. SPI通信原理

SPI是一种同步串行通信接口,由主设备产生时钟信号,通过MOSI线发送数据给从设备,同时通过MISO线接收从设备的数据。

2. W25Q64 Flash存储器特性

  • 容量:8Mbit
  • 组织:1M字节,256字节/页,4K字节/扇区
  • 接口:SPI
  • 速度:最高104MHz

3. 硬件连接

  • W25Q64_CS 连接到 STM32 GPIO(如PA4)
  • W25Q64_SCK 连接到 STM32 SPI SCK(如PA5)
  • W25Q64_MOSI 连接到 STM32 SPI MOSI(如PA7)
  • W25Q64_MISO 连接到 STM32 SPI MISO(如PA6)
  • VCC 连接到 3.3V
  • GND 连接到 GND

4. SPI初始化

4.1 硬件SPI初始化

硬件SPI初始化涉及到SPI外设的配置,包括时钟、数据大小、主从模式等。

void SPI_Hardware_Init(void) {
    GPIO_InitTypeDef GPIO_InitStructure;
    SPI_InitTypeDef SPI_InitStructure;

    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);
    RCC_APB1PeriphClockCmd(RCC_APB1Periph_SPI1, ENABLE);

    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_5 | GPIO_Pin_6 | GPIO_Pin_7;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_Init(GPIOA, &GPIO_InitStructure);

    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_4;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
    GPIO_Init(GPIOA, &GPIO_InitStructure);

    SPI_InitStructure.SPI_Direction = SPI_Direction_2Lines_FullDuplex;
    SPI_InitStructure.SPI_Mode = SPI_Mode_Master;
    SPI_InitStructure.SPI_DataSize = SPI_DataSize_8b;
    SPI_InitStructure.SPI_CPOL = SPI_CPOL_High;
    SPI_InitStructure.SPI_CPHA = SPI_CPHA_2Edge;
    SPI_InitStructure.SPI_NSS = SPI_NSS_Soft;
    SPI_InitStructure.SPI_BaudRatePrescaler = SPI_BaudRatePrescaler_16;
    SPI_InitStructure.SPI_FirstBit = SPI_FirstBit_MSB;
    SPI_Init(SPI1, &SPI_InitStructure);

    SPI_Cmd(SPI1, ENABLE);
}
4.2 模拟SPI初始化

模拟SPI通过直接操作GPIO引脚来模拟SPI通信。

void SPI_Software_Init(void) {
    GPIO_InitTypeDef GPIO_InitStructure;

    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);

    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_5 | GPIO_Pin_7;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_Init(GPIOA, &GPIO_InitStructure);

    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU; // 输入上拉模式
    GPIO_Init(GPIOA, &GPIO_InitStructure);
}

5. W25Q64操作

5.1 W25Q64初始化
void W25Q64_Init(void) {
    SPI_Hardware_Init(); // 硬件SPI初始化
    SPI_Software_Init(); // 模拟SPI初始化
}
5.2 读取W25Q64 ID
uint16_t W25Q64_ReadID(void) {
    uint16_t id = 0;
    SPI_CS_Low(); // 片选
    SPI_SendByte(0x90); // 发送读ID指令
    id = SPI_ReceiveByte() << 8; // 读取制造商ID
    id |= SPI_ReceiveByte(); // 读取设备ID
    SPI_CS_High(); // 取消片选
    return id;
}
5.3 写入W25Q64
void W25Q64_WriteByte(uint32_t addr, uint8_t data) {
    SPI_CS_Low(); // 片选
    SPI_SendByte(0x02); // 发送页编程指令
    SPI_SendByte(addr >> 16); // 发送地址的高8位
    SPI_SendByte(addr >> 8);  // 发送地址的中8位
    SPI_SendByte(addr);       // 发送地址的低8位
    SPI_SendByte(data);       // 发送数据
    SPI_CS_High(); // 取消片选
    // 等待写入完成...
}
5.4 从W25Q64读取数据
uint8_t W25Q64_ReadByte(uint32_t addr) {
    uint8_t data;
    SPI_CS_Low(); // 片选
    SPI_SendByte(0x03); // 发送读数据指令
    SPI_SendByte(addr >> 16); // 发送地址的高8位
    SPI_SendByte(addr >> 8);  // 发送地址的中8位
    SPI_SendByte(addr);       // 发送地址的低8位
    data = SPI_ReceiveByte(); // 读取数据
    SPI_CS_High(); // 取消片选
    return data;
}

6. 完整示例代码

以下是使用硬件SPI和模拟SPI驱动W25Q64芯片的完整示例代码。

int main(void) {
    // 系统初始化
    SystemInit();

    // 初始化W25Q64
    W25Q64_Init();

    // 读取W25Q64 ID
    uint16_t id = W25Q64_ReadID();
    printf("W25Q64 ID: 0x%X\r\n", id);

    // 使用硬件SPI写入数据
    W25Q64_WriteByte(0x000000, 0xAA);

    // 延迟一段时间,等待数据写入完成
    for (volatile int i = 0; i < 0x100000; i++);

    // 使用模拟SPI读取数据
    uint8_t data = W25Q64_ReadByte(0x000000);
    printf("Read data: 0x%X\r\n", data);

    while(1) {
        // 其他应用代码
    }
}

7. 结论

本文详细介绍了STM32微控制器通过硬件SPI和模拟SPI实现W25Q64 Flash存储器的读写操作。通过示例代码,读者可以快速理解和应用SPI通信。在实际开发中,开发者可能需要根据具体的硬件配置和需求进行适当的调整和优化。

请注意,示例代码需要根据实际的硬件配置和开发环境进行适配。

✅作者简介:热爱科研的嵌入式开发者,修心和技术同步精进

❤欢迎关注我的知乎:对error视而不见

代码获取、问题探讨及文章转载可私信。

☁ 愿你的生命中有够多的云翳,来造就一个美丽的黄昏。

🍎获取更多嵌入式资料可点击链接进群领取,谢谢支持!👇

点击领取更多详细资料

  • 9
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
下面是基于stm32f10c8t6硬件spi通信的w25q64的驱动及测试代码: 驱动代码: #include "stm32f10x.h" #define W25Q64_CS_PIN GPIO_Pin_4 #define W25Q64_CS_PORT GPIOA void W25Q64_Init(void) { GPIO_InitTypeDef GPIO_InitStructure; RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE); GPIO_InitStructure.GPIO_Pin = W25Q64_CS_PIN; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_Init(W25Q64_CS_PORT, &GPIO_InitStructure); GPIO_SetBits(W25Q64_CS_PORT, W25Q64_CS_PIN); } void W25Q64_ChipSelect(uint8_t select) { if (select == 0) { GPIO_ResetBits(W25Q64_CS_PORT, W25Q64_CS_PIN); } else { GPIO_SetBits(W25Q64_CS_PORT, W25Q64_CS_PIN); } } uint8_t W25Q64_SendByte(uint8_t txData) { while (SPI_I2S_GetFlagStatus(SPI1, SPI_I2S_FLAG_TXE) == RESET); SPI_I2S_SendData(SPI1, txData); while (SPI_I2S_GetFlagStatus(SPI1, SPI_I2S_FLAG_RXNE) == RESET); return SPI_I2S_ReceiveData(SPI1); } void W25Q64_WriteEnable(void) { W25Q64_ChipSelect(0); W25Q64_SendByte(0x06); W25Q64_ChipSelect(1); } void W25Q64_WriteDisable(void) { W25Q64_ChipSelect(0); W25Q64_SendByte(0x04); W25Q64_ChipSelect(1); } void W25Q64_EraseSector(uint32_t sectorAddr) { W25Q64_WriteEnable(); W25Q64_ChipSelect(0); W25Q64_SendByte(0x20); W25Q64_SendByte((sectorAddr>>16) & 0xff); W25Q64_SendByte((sectorAddr>>8) & 0xff); W25Q64_SendByte(sectorAddr & 0xff); W25Q64_ChipSelect(1); while ((W25Q64_ReadStatus() & 0x01) == 0x01); } void W25Q64_WritePage(uint32_t pageAddr, uint8_t *pData, uint32_t size) { W25Q64_WriteEnable(); W25Q64_ChipSelect(0); W25Q64_SendByte(0x02); W25Q64_SendByte((pageAddr>>16) & 0xff); W25Q64_SendByte((pageAddr>>8) & 0xff); W25Q64_SendByte(pageAddr & 0xff); while (size--) { W25Q64_SendByte(*pData++); } W25Q64_ChipSelect(1); while ((W25Q64_ReadStatus() & 0x01) == 0x01); } void W25Q64_ReadData(uint32_t addr, uint8_t *pData, uint32_t size) { W25Q64_ChipSelect(0); W25Q64_SendByte(0x03); W25Q64_SendByte((addr>>16) & 0xff); W25Q64_SendByte((addr>>8) & 0xff); W25Q64_SendByte(addr & 0xff); while (size--) { *pData++ = W25Q64_SendByte(0xff); } W25Q64_ChipSelect(1); } uint8_t W25Q64_ReadStatus(void) { uint8_t status; W25Q64_ChipSelect(0); W25Q64_SendByte(0x05); status = W25Q64_SendByte(0xff); W25Q64_ChipSelect(1); return status; } void W25Q64_WriteStatus(uint8_t status) { W25Q64_WriteEnable(); W25Q64_ChipSelect(0); W25Q64_SendByte(0x01); W25Q64_SendByte(status); W25Q64_ChipSelect(1); while ((W25Q64_ReadStatus() & 0x01) == 0x01); } 测试代码: #include "stm32f10x.h" #include "w25q64.h" int main(void) { uint8_t data[256]; uint32_t i; uint8_t status; RCC_APB2PeriphClockCmd(RCC_APB2Periph_SPI1, ENABLE); SPI_InitTypeDef SPI_InitStructure; SPI_InitStructure.SPI_Direction = SPI_Direction_2Lines_FullDuplex; SPI_InitStructure.SPI_Mode = SPI_Mode_Master; SPI_InitStructure.SPI_DataSize = SPI_DataSize_8b; SPI_InitStructure.SPI_CPOL = SPI_CPOL_High; SPI_InitStructure.SPI_CPHA = SPI_CPHA_2Edge; SPI_InitStructure.SPI_NSS = SPI_NSS_Soft; SPI_InitStructure.SPI_BaudRatePrescaler = SPI_BaudRatePrescaler_16; SPI_InitStructure.SPI_FirstBit = SPI_FirstBit_MSB; SPI_InitStructure.SPI_CRCPolynomial = 7; SPI_Init(SPI1, &SPI_InitStructure); SPI_Cmd(SPI1, ENABLE); W25Q64_Init(); W25Q64_EraseSector(0x000000); for (i = 0; i < 256; i++) { data[i] = i; } W25Q64_WritePage(0x000000, data, 256); W25Q64_ReadData(0x000000, data, 256); for (i = 0; i < 256; i++) { if (data[i] != i) { while (1); } } status = W25Q64_ReadStatus(); status &= ~0x02; status |= 0x02; W25Q64_WriteStatus(status); while (1); }

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

嵌入式杂谈

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值