sdio接口_【STM32Cube-19】使用SDMMC接口读写SD卡数据

2129e3ccb307cb8725f38f5f5996f191.png

寻求更好的阅读体验,请移步Mculover666的个人博客:

【STM32Cube-19】使用SDMMC接口读写SD卡数据​www.mculover666.cn
9c3454446872545e9036575a80a6a7f7.png

本篇详细的记录了如何使用STM32CubeMX配置STM32L431RCT6的硬件SDMMC外设读取SD卡数据。


1. 准备工作

硬件准备

  • 开发板 首先需要准备一个开发板,这里我准备的是STM32L4的开发板(BearPi):

a91122505828e23872e119d4e258ba82.png
  • Micro SD卡 小熊派开发板板载 Micro SD 卡槽,最大支持 32 GB,需要提前自行准备一张 Micro SD卡,如图:

6b422d9d5b167c01a8679df2f80a9acd.png

软件准备

  • 需要安装好Keil - MDK及芯片对应的包,以便编译和下载生成的代码;
  • 准备一个串口调试助手,这里我使用的是Serial Port Utility
Keil MDK和串口助手Serial Port Utility 的安装包都可以 在文末关注公众号获取,回复关键字获取相应的安装包:

503aaa23b2e461deb8624453030fd582.png

2.生成MDK工程

选择芯片型号

打开STM32CubeMX,打开MCU选择器:

38908257fe15e4a0213dcecca451a513.png

搜索并选中芯片STM32L431RCT6:

0856d04eaa5c82294ecdf4beb4166d44.png

配置时钟源

  • 如果选择使用外部高速时钟(HSE),则需要在System Core中配置RCC;
  • 如果使用默认内部时钟(HSI),这一步可以略过;

这里我都使用外部时钟:

63fa0d961b33176cc055c2e6530a87cf.png

配置串口

小熊派开发板板载ST-Link并且虚拟了一个串口,原理图如下:

fbeb2d24cecc045c977a847df4ce3b0e.png

这里我将开关拨到AT-MCU模式,使PC的串口与USART1之间连接。

接下来开始配置USART1

855c786568e8d62c0b8ca13a91d6ad13.png

配置 SDMMC 接口

知识小卡片 —— SDMMC接口

SDMMC接口的全称叫SD/SDIO MMC card host interface,SD/SDIO MMC 卡 主机接口,通俗的来说,就是这个接口支持SD卡,支持SDIO设备,支持MMC卡。

知识小卡片结束啦~

首先查看小熊派开发板的原理图:

f092ac44f7adb7b1f118883ddb0bb211.png

然后根据原理图配置 SDMMC 接口:

d1c02ef2d18924169760b5d423f3ddb5.png

配置时钟树

STM32L4的最高主频到80M,所以配置PLL,最后使HCLK = 80Mhz即可:

e939105ee18495416d3f4c5fdace14e0.png

生成工程设置

e278f82665b967656419de006b589560.png

代码生成设置

最后设置生成独立的初始化文件:

8c69f5ed482e7467ed1ace5f1a5abf15.png

生成代码

点击GENERATE CODE即可生成MDK-V5工程:

f28b40b7c02589969443f5a36605c6aa.png

3. 在MDK中编写、编译、下载用户代码

重定向printf( )函数

参考:【STM32Cube_09】重定向printf函数到串口输出的多种方法。

读取SD卡信息并打印

SD 卡系统(包括主机和 SD 卡)定义了两种操作模式:

  • 卡识别模式
  • 数据传输模式

系统复位后,主机处于卡识别模式,寻找总线上可用的 SD卡设备;同时,SD 卡也处于卡 识别模式,直到被主机识别到。

使用STM32CubeMX初始化的工程中会自动生成 SDMMC 初始化函数,向 SD 卡发送命令,当 SD 卡接收到命令后, SD 卡就会进入数据传输模式,而主机在总线上所有卡被识别后也进入数据传输模式

所以在操作之前,需要先检查 SD 卡是否处于数据传输模式并且处于数据传输状态:

main函数中首先定义一个变量用于存储 SD 卡状态:

int sdcard_status = 0;
HAL_SD_CardCIDTypeDef sdcard_cid;

然后在while(1)之前编写如下读取信息代码:

/* USER CODE BEGIN 2 */
printf("Micro SD Card Test...rn");

/* 检测SD卡是否正常(处于数据传输模式的传输状态) */
sdcard_status = HAL_SD_GetCardState(&hsd1);
if(sdcard_status == HAL_SD_CARD_TRANSFER)
{
    printf("SD card init ok!rnrn");

    //打印SD卡基本信息
    printf("SD card information!rn");
    printf("CardCapacity: %llurn",((unsigned long long)hsd1.SdCard.BlockSize*hsd1.SdCard.BlockNbr));
    printf("CardBlockSize: %d rn",hsd1.SdCard.BlockSize);
    printf("RCA: %d rn",hsd1.SdCard.RelCardAdd);
    printf("CardType: %d rn",hsd1.SdCard.CardType);

    //读取并打印SD卡的CID信息
    HAL_SD_GetCardCID(&hsd1,&sdcard_cid);
    printf("ManufacturerID: %d rn",sdcard_cid.ManufacturerID);
}
else
{ 
    printf("SD card init fail!rn" );
    return 0;
}
/* USER CODE END 2 */

编译下载后串口助手输出结果如下:

dcfa5107cd6966272caaccf59a7fa873.png

擦除SD卡块数据

为了验证实验的正确性或,先擦除数据:

/* 擦除SD卡块 */
printf("------------------- Block Erase -------------------------------rn");
sdcard_status = HAL_SD_Erase(&hsd1, 0, 512);
if (sdcard_status == 0)
{
    printf("Erase block okrn");
}
else
{
    printf("Erase block failrn");
}

读取SD卡块数据

首先开辟一个全局缓冲区,用于存放从SD卡读出的数据:

/* Private user code ---------------------------------------------------------*/
/* USER CODE BEGIN 0 */
uint8_t read_buf[512];
/* USER CODE END 0 */

然后在之前读取信息的代码之后添加读取数据的代码:

/* 读取未操作之前的数据 */
printf("------------------- Read SD card block data Test ------------------rn");
sdcard_status = HAL_SD_ReadBlocks(&hsd1,(uint8_t *)read_buf,0,1,0xffff);
if(sdcard_status == 0)
{ 
    printf("Read block data ok rn" );
    for(i = 0; i < 512; i++)
    {
        printf("0x%02x ", read_buf[i]);
        if((i+1)%16 == 0)
        {
            printf("rn");
        }
    }
}
else
{
    printf("Read block data fail!rn " );
}

向SD卡块写入数据

同样的,开辟一个全局缓冲区,用于存放即将要写入SD卡的数据:

uint8_t write_buf[512];

然后在之前读取数据的代码之后添加的代码,将缓冲区的数据赋初值:

/* 填充缓冲区数据 */
for(i = 0; i < 512; i++)
{
    write_buf[i] = i % 256;
}

然后继续添加代码,将该缓冲区数据写入SD卡:

/* 向SD卡块写入数据 */
printf("------------------- Write SD card block data Test ------------------rn");
sdcard_status = HAL_SD_WriteBlocks(&hsd1,(uint8_t *)write_buf,0,1,0xffff);
if(sdcard_status == 0)
{ 
    printf("Write block data ok rn" );
}
else
{
    printf("Write block data fail!rn " );
}

添加完之后,为了检查数据是否正常写入,再将数据读出:

/* 读取操作之后的数据 */
printf("------------------- Read SD card block data after Write ------------------rn");
sdcard_status = HAL_SD_ReadBlocks(&hsd1,(uint8_t *)read_buf,0,1,0xffff);
if(sdcard_status == 0)
{ 
    printf("Read block data ok rn" );
    for(i = 0; i < 512; i++)
    {
        printf("0x%02x ", read_buf[i]);
        if((i+1)%16 == 0)
        {
            printf("rn");
        }
    }
}

将程序编译下载,最终的实验结果如下:

4a1bf84872dec9f0184d798f71e38bbb.png

3226b0d8efae121209d1da5dcb5eb323.png

a7e9ff6b74a7504a5a305bda0a1bd33f.png

0c55003729c334d486e3df299097a92a.png

9820819b599c2f77df76bdbd89675cbc.png

至此,我们已经学会如何使用硬件SDMMC接口读取SD数据,STM32CubeMX系列教程完结。

更多精彩文章及资源,请关注我的微信公众号:『mculover666』。

48e5b90af40cefa52d122c9c5c75b3d4.png
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值