| **银杏科技有限公司旗下技术文档发布平台** ||||
|技术支持电话|**0379-69926675-801**|||
|技术支持邮件|Gingko@vip.163.com|||
^ 版本 ^ 日期 ^ 作者 ^ 修改内容 ^
| V1.0 | 2020-02-28 | gingko | 初次建立 |
\\
\\
\\
\\
\\
\\
===== STM32CubeMX教程十六——SDIO实验 =====
\\
\\
1.在主界面选择File-->New Project 或者直接点击ACCEE TO MCU SELECTOR
{{ :icore4t:icore4t_cube_16_1.png?direct |}}
2.出现芯片型号选择,搜索自己芯片的型号,双击型号,或者点击Start Project进入配置
在搜索栏的下面,提供的各 种查找方式,可以选择芯片内核,型号,等等,可以帮助你查找芯片。本实验选取的芯片型号为:STM32H750IBKx。
{{ :icore4t:icore4t_cube_16_2.png?direct |}}
3.配置RCC,使用外部时钟源
{{ :icore4t:icore4t_cube_16_3.png?direct |}}
4.时基源选择SysTick
{{ :icore4t:icore4t_cube_16_4.png?direct |}}
5.将PA10,PB7,PB8设置为GPIO_Output
{{ :icore4t:icore4t_cube_16_5.png?direct |}}
6.引脚模式配置
{{ :icore4t:icore4t_cube_16_6.png?direct |}}
{{ :icore4t:icore4t_cube_16_7.png?direct |}}
7.设置串口
{{ :icore4t:icore4t_cube_16_8.png?direct |}}
8.在NVIC Settings一栏使能接收中断
{{ :icore4t:icore4t_cube_16_9.png?direct |}}
9.配置SDMMC1
{{ :icore4t:icore4t_cube_16_10.png?direct |}}
10.时钟源设置,选择外部高速时钟源,配置为最大主频
{{ :icore4t:icore4t_cube_16_11.png?direct |}}
{{ :icore4t:icore4t_cube_16_12.png?direct |}}
11.工程文件的设置, 这里就是工程的各种配置 我们只用到有限几个,其他的默认即可 IDE我们使用的是 MDK V5.27
{{ :icore4t:icore4t_cube_16_13.png?direct |}}
12.点击Code Generator,进行进一步配置
{{ :icore4t:icore4t_cube_16_14.png?direct |}}
* **Copy all used libraries into the project folder**
* **将HAL库的所有.C和.H都复制到所建工程中**
* 优点:这样如果后续需要新增其他外设又可能不再用STM32CubeMX的时候便会很方便
* 缺点:体积大,编译时间很长
* **Copy only the necessary library files**
* **只复制所需要的.C和.H(推荐)**
* 优点:体积相对小,编译时间短,并且工程可复制拷贝
* 缺点:新增外设时需要重新用STM32CubeMX导入
* **Add necessary library files as reference in the toolchain project configuration file**
* **不复制文件,直接从软件包存放位置导入.C和.H**
* 优点:体积小,比较节约硬盘空间
* 缺点:复制到其他电脑上或者软件包位置改变,就需要修改相对应的路径
* 自行选择方式即可
13.然后点击GENERATE CODE 创建工程
{{ :icore4t:icore4t_cube_16_15.png?direct |}}
创建成功,打开工程。
\\
\\
===== 实验十六:SDIO实验——读取SD卡信息 =====
==== 一、 实验目的与意义 ====
- 了解STM32 SDIO结构。
- 了解STM32 SDIO特征。
- 掌握SDIO的使用方法。
- 掌握STM32 HAL库中SDIO属性的配置方法。
- 掌握KEILMDK 集成开发环境使用方法。
==== 二、 实验设备及平台 ====
- iCore4T 双核心板。[[https://item.taobao.com/item.htm?spm=a1z10.1-c.w137644-251734891.3.5923532fDrMDOe&id=610595120319|点击购买]]
- JLINK(或相同功能)仿真器。[[https://item.taobao.com/item.htm?id=554869837940|点击购买]]
- Micro USB线缆。
- Keil MDK 开发平台。
- STM32CubeMX开发平台。
- 装有WIN XP(及更高版本)系统的计算机。
==== 三、 实验原理 ====
=== 1.SDIO的定义 ===
* SDIO在SD标准上定义了一种外设接口。目前,SDIO主要有两类应用——可移动和不可移动。可移动设备作为Palm和Windows Mobile的扩展设备,用来增加蓝牙、照相机、GPS和802.11b功能。不可移动设备遵循相同的电气标准,但不要求符合物理标准。某些手机内包含通过SDIO连接CPU的802.11芯片。此举将“珍贵”的I/ O管脚资源用于更重要的功能。
* 蓝牙、照相机、GPS和802.11b设备有专为它们定义的应用规范。这些应用规范与为PCI和USB设备定义的类规范很相像。它们允许任何宿主设备与任意外设“通话”,只要它们都支持应用规范。
* SDIO和SD卡规范间的一个重要区别是增加了低速标准。SDIO卡只需要SPI和1位SD传输模式。低速卡的目标应用是以最小的硬件开支支持低速I/ O能力。低速卡支持类似调制解调器、条码扫描仪和GPS接受器等应用。对“组合”卡(存储器+ SDIO)而言,全速和4位操作对卡内存储器和SDIO部分都是强制要求的。
=== 2.SDIO的功能 ===
* 转换设备的能力 SD标准有很大的弹性,比方说,SD插槽不仅能插入内存卡。SDIO卡是一种界面,即由使用标准SD卡插槽来扩展设备的功能,让设备拥有新的功能。
* 新功能列举如下:GPS 相机 Wi-Fi 调频广播 以太网 条形码读卡器 蓝牙; 将与SD memory card同样的终端形状的卡槽利用于扩张卡的规格。把对应SDIO卡槽(插口)称为SDIO卡,扩张卡称为SDIO卡。
* 为了在个人电脑和PDA等中添加功能而使用SD卡的规格,由于可以使用比PC卡形状还小的卡和插槽,相比于笔记本电脑,更多采用于小巧的PDA中。主要销售的产品有,面向PDA的PHS通讯卡和蓝牙通讯卡,无线网卡,小型数码相机等。
* 为了使用SDIO,必须要有对应的插槽,数码相机等memory card的专用缝儿插槽不对应SDIO卡。SDIO插槽中能够插入SD memory card来读写。
* SD/SDIO MMC卡主机接口(SDIO)提供APB2外设总线与多媒体卡(MMC)、SD卡、SDIO卡以及CE-ATA设备之间的接口。
=== 3.SDIO具有如下主要特性: ===
* 完全兼容多媒体卡系统规范版本4.2。卡支持三种不同数据总线模式:1位(默认)、4位和8位。
* 完全兼容先前版本的多媒体卡(向前兼容性)。
* 完全兼容SD存储卡规范版本2.0。
* 完全兼容SDI/O卡规范版本2.0:卡支持两种不同数据总线模式:1位(默认)和4位。
* 完全支持CE-ATA功能(完全符合CE-ATA数字协议版本1.1)。
* 对于8位模式,数据传输高达48MHz。
* 数据和命令输出使能信号,控制外部双向驱动程序。
=== 4.SD卡识别过程: ===
* 多媒体卡和SD卡的卡识别过程有所不同。对于多媒体,识别过程是以时钟速率开始。SDIO_CMD线输出驱动器是开漏引脚,在此识别过程中允许并行的卡操作。对准过程以如下方式完成:
* (1)激活总线。
* (2)SDIO卡主机广播SEND_OP_COND(CMD1)以接收运行条件。
* (3)卡以其运行条件寄存器的内容进行响应。
* (4)不兼容的卡将被置于无效状态。
* (5)SDIO卡主机向所有有效广播ALL_SEND_CID(CMD2)。
* (6)这些卡将发回其唯一的卡识别号(CID)并进入识别状态。
* (7)SDIO卡主机向某个地址的有效卡发出SET_RELATIVE_ADDR(CMD3)。这一新地址称为相对卡地址(RCA);它比CID更短,可对卡进行寻址。分配的卡变为待机状态。SDIO卡主机可以重新发出此命令以更改RCA。卡的RCA是最后分配的值。
* (8)SDIO卡主机对所有有效卡重复步骤5到7。
=== 5.SDIO接口的设备 ===
{{ :icore4t:icore4t_arm_hal_16_1.png?direct |}}
=== 6.SD卡物理结构 ===
{{ :icore4t:icore4t_arm_hal_16_2.png?direct |}}
{{ :icore4t:icore4t_arm_hal_16_3.png?direct |}}
=== 7.原理图 ===
{{ :icore4t:icore4t_arm_hal_16_4.png?direct |}}
==== 四、 实验程序 ====
=== 1.主函数 ===
int main(void)
{
int i,j;
unsigned char write_buffer[512];
unsigned char read_buffer[512];
int error = 0;
HAL_Init();
SystemClock_Config();
i2c.initialize();
axp152.initialize();
axp152.set_dcdc1(3500);//[ARM & FPGA BK1/2/6 &OTHER]
axp152.set_dcdc2(1200);//[FPGA INT & PLL D]
axp152.set_aldo1(2500);//[FPGA PLL A]
axp152.set_dcdc4(3300);//[POWER_OUTPUT]
axp152.set_dcdc3(3300);//[FPGA BK4][Adjustable]
axp152.set_aldo2(3300);//[FPGA BK3][Adjustable]
axp152.set_dldo1(3300);//[FPGA BK7][Adjustable]
axp152.set_dldo2(3300);//[FPGA BK5][Adjustable]
MX_GPIO_Init();
MX_USART2_UART_Init();
MX_SDMMC1_SD_Init();
usart2.initialize(115200); //串口波特率设置
usart2.printf("\x0c"); //清屏
usart2.printf("\033[1;32;40m"); //设置终端字体为绿色
usart2.printf("Hello,I am iCore4T!\r\n\r\n");
// 配置 SD 接口位宽为 4bit 用于数据传输
if(HAL_SD_ConfigWideBusOperation(&hsd1, SDMMC_BUS_WIDE_4B) != HAL_OK){
usart2.printf("SD Card Error!\r\n");
while(1){
LED_ON;
HAL_Delay(500);
LED_OFF;
HAL_Delay(500);
}
}else{
usart2.printf("SD Card OK!\r\n");
}
//打印SD卡类型
switch(hsd1.SdCard.CardType){
case CARD_SDSC:
usart2.printf("SD CardType\t\t: CARD_SDSC\r\n");
break;
case CARD_SDHC_SDXC:
usart2.printf("SD CardType\t\t: CARD_SDHC_SDXC\r\n");
break;
case CARD_SECURED:
usart2.printf("SD CardType\t\t: CARD_SECURED\r\n");
break;
}
switch(hsd1.SdCard.CardVersion){ //打印SD卡规范版本
case CARD_V1_X:
usart2.printf("SD CardVersion\t: CARD_V1_X\r\n");
break;
case CARD_V2_X:
usart2.printf("SD SD CardVersion\t: CARD_V2_X\r\n");
break;
}
usart2.printf("SD CardCapacity\t\t: %dMB\r\n",(unsigned int)(hsd1.SdCard.BlockNbr * hsd1.SdCard.BlockSize) >> 20); //打印SD卡容量
usart2.printf("SD BlockSize\t\t: %dByte\r\n",hsd1.SdCard.BlockSize); //打印SD卡数据块大小
usart2.printf("SD RelCardAdd\t\t: %d\r\n",hsd1.SdCard.RelCardAdd); //打印SD卡相对地址
//从缓冲区写入512字节数据到SD卡数据块
for(i = 0;i < 512;i++){
write_buffer[i] = i % 256;
}
HAL_SD_WriteBlocks(&hsd1,write_buffer,0,1,1000);
//读取SD卡数据块数据到缓冲区
HAL_SD_ReadBlocks(&hsd1,read_buffer,0,1,1000);
for(i = 0;i < 512;i ++){
if(write_buffer[i] != read_buffer[i]){ //比较数据是否读写正确
error ++;
}
}
if(error){ //测试失败,LED闪烁
usart2.printf("SD TEST FAIL!\r\n");
while(1){
LED_ON;
HAL_Delay(500);
LED_OFF;
HAL_Delay(500);
}
}else{ //测试成功打印读取到的数据
LED_ON;
usart2.printf("SD TEST OK!\r\n");
usart2.printf("read data:\r\n");
for(i = 0;i < 32;i++){
for(j = 0; j < 16; j ++)
usart2.printf(" %02X",read_buffer[i*16+j]);
usart2.printf("\r\n");
}
}
while (1)
{
}
}
=== 2.SDMMC初始化函数 ===
//SDMMC初始化
void MX_SDMMC1_SD_Init(void)
{
hsd1.Instance = SDMMC1;
hsd1.Init.ClockEdge = SDMMC_CLOCK_EDGE_RISING; //上升沿
hsd1.Init.ClockPowerSave = SDMMC_CLOCK_POWER_SAVE_DISABLE; //空闲时不关闭时钟电源
hsd1.Init.BusWide = SDMMC_BUS_WIDE_4B; //4 位数据线
hsd1.Init.HardwareFlowControl = SDMMC_HARDWARE_FLOW_CONTROL_DISABLE; //关闭硬件流控
hsd1.Init.ClockDiv = 10; //分频
hsd1.Init.TranceiverPresent = SDMMC_TRANSCEIVER_NOT_PRESENT;
if (HAL_SD_Init(&hsd1) != HAL_OK)
{
Error_Handler();
}
}
=== 3.SD卡写数据块函数 ===
HAL_SD_WriteBlocks(SD_HandleTypeDef *hsd, uint8_t *pData, uint32_t BlockAdd, uint32_t NumberOfBlocks, uint32_t Timeout)
//*hsd:指向SD卡的指针
//*pData:指向要写入的数据的指针
//BlockAdd:数据块地址
//NumberOfBlocks:写入的块数
//Timeout:写入超时设置
=== 4.SD读数据块函数 ===
HAL_SD_ReadBlocks(SD_HandleTypeDef *hsd, uint8_t *pData, uint32_t BlockAdd, uint32_t NumberOfBlocks, uint32_t Timeout)
//*hsd:指向SD卡的指针
//*pData:指向数据读取后存放区的指针
//BlockAdd:数据块地址
//NumberOfBlocks:读取的块数
//Timeout:读取超时设置
=== 5.SDMMC外设管理结构体 ===
typedef struct {
SD_TypeDef *Instance; /*!< SDMMC 寄存器基地址*/
SD_InitTypeDef Init; /*!< SD 初始化结构体*/
HAL_LockTypeDef Lock; /*!< SD 锁资源*/
uint32_t *pTxBuffPtr; /*!< 存放发送数据地址的指针*/
uint32_t TxXferSize; /*!< 发送数据的大小 */
uint32_t *pRxBuffPtr; /*!< 存放接受数据地址的指针*/
uint32_t RxXferSize; /*!< 接受数据的大小*/
__IO uint32_t Context; /*!< SDMMC 的工作模式 */
__IO HAL_SD_StateTypeDef State; /*!< SD 卡的状态值*/
__IO uint32_t ErrorCode; /*!< SD 错误操作返回值*/
HAL_SD_CardInfoTypeDef SdCard; /*!< SD 卡的信息*/
uint32_t CSD[4]; /*!< SD 卡的 CSD 寄存器值*/
uint32_t CID[4]; /*!< SD 卡的 CID 寄存器值*/
} SD_HandleTypeDef;
=== 6.SDMMC 数据初始化结构体 ===
typedef struct {
uint32_t DataTimeOut; // 数据传输超时
uint32_t DataLength; // 数据长度
uint32_t DataBlockSize; // 数据块大小
uint32_t TransferDir; // 数据传输方向
uint32_t TransferMode; // 数据传输模式
uint32_t DPSM; // 数据路径状态机
} SDMMC_DataInitTypeDef;
=== 7.SD卡信息结构体 ===
typedef struct
{
uint32_t CardType; /*卡种类*/
uint32_t CardVersion; /*卡版本*/
uint32_t Class; /*卡类*/
uint32_t RelCardAdd; /*相对卡地址*/
uint32_t BlockNbr; /*以块为单位指定卡容量*/
uint32_t BlockSize; /*一个块大小(以字节为单位)*/
uint32_t LogBlockNbr; /*以块为单位指定卡逻辑容量*/
uint32_t LogBlockSize; /*逻辑块大小(以字节为单位)*/
uint32_t CardSpeed; /*卡速度*/
}HAL_SD_CardInfoTypeDef;
==== 五、 实验步骤 ====
- 把仿真器与iCore4T的SWD调试口相连(直接相连或者通过转接器相连);
- 把iCore4T通过Micro USB线与计算机相连,为iCore4T供电;
- 打开PuTTY串口终端;
- 打开Keil MDK 开发环境,并打开本实验工程;
- 烧写程序到iCore4T上;
- 也可以进入Debug 模式,单步运行或设置断点验证程序逻辑。
==== 六、 实验现象 ====
终端显示出SD卡的相关信息,并输出写入SD卡的数据。
{{ :icore4t:icore4t_arm_hal_16_5.png?direct |}}