这里会显示出您选择的修订版和当前版本之间的差别。
后一修订版 | 前一修订版 | ||
icore3l_arm_16 [2020/11/18 10:19] zgf 创建 |
icore3l_arm_16 [2022/03/19 11:01] (当前版本) sean |
||
---|---|---|---|
行 2: | 行 2: | ||
|技术支持电话|**0379-69926675-801**||| | |技术支持电话|**0379-69926675-801**||| | ||
|技术支持邮件|Gingko@vip.163.com||| | |技术支持邮件|Gingko@vip.163.com||| | ||
- | |技术论坛|http://www.eeschool.org||| | ||
^ 版本 ^ 日期 ^ 作者 ^ 修改内容 ^ | ^ 版本 ^ 日期 ^ 作者 ^ 修改内容 ^ | ||
| V1.0 | 2020-11-18 | gingko | 初次建立 | | | V1.0 | 2020-11-18 | gingko | 初次建立 | | ||
===== STM32CubeMX教程十六——SDIO实验 ===== | ===== STM32CubeMX教程十六——SDIO实验 ===== | ||
1.在主界面选择File-->New Project或者直接点击ACCEE TO MCU SELECTOR | 1.在主界面选择File-->New Project或者直接点击ACCEE TO MCU SELECTOR | ||
+ | {{ :icore3l:icore3l_cube_16_1.png?direct |}} | ||
+ | 2.出现芯片型号选择,搜索自己芯片的型号,双击型号,或者点击Start Project进入配置在搜索栏的下面,提供的各种查找方式,可以选择芯片内核,型号,等等,可以帮助你查找芯片。本实验选取的芯片型号为:STM32FIGHx。 | ||
+ | {{ :icore3l:icore3l_cube_16_2.png?direct |}} | ||
+ | 3.配置RCC,使用外部时钟源 | ||
+ | {{ :icore3l:icore3l_cube_16_3.png?direct |}} | ||
+ | 4.Debug选择Serial Wire,时基源选择SysTick | ||
+ | {{ :icore3l:icore3l_cube_16_4.png?direct |}} | ||
+ | {{ :icore3l:icore3l_cube_16_5.png?direct |}} | ||
+ | 5.将PI3,PI4,PH14设置为GPIO_Output | ||
+ | {{ :icore3l:icore3l_cube_16_6.png?direct |}} | ||
+ | 6.引脚模式配置 | ||
+ | {{ :icore3l:icore3l_cube_16_7.png?direct |}} | ||
+ | 7.设置串口 | ||
+ | {{ :icore3l:icore3l_cube_16_8.png?direct |}} | ||
+ | 8.在NVIC Settings一栏使能接收中断 | ||
+ | {{ :icore3l:icore3l_cube_16_9.png?direct |}} | ||
+ | 9.配置SDIO | ||
+ | {{ :icore3l:icore3l_cube_16_10.png?direct |}} | ||
+ | 10.时钟源设置,选择外部高速时钟源,配置为最大主频 | ||
+ | {{ :icore3l:icore3l_cube_16_11.png?direct |}} | ||
+ | 11.工程文件的设置, 这里就是工程的各种配置 我们只用到有限几个,其他的默认即可IDE我们使用的是MDK V5.27 | ||
+ | {{ :icore3l:icore3l_cube_16_12.png?direct |}} | ||
+ | 12.点击Code Generator,进行进一步配置 | ||
+ | {{ :icore3l:icore3l_cube_16_13.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创建工程 | ||
+ | {{ :icore3l:icore3l_cube_16_14.png?direct |}} | ||
+ | 创建成功,打开工程。 | ||
+ | \\ | ||
+ | \\ | ||
+ | \\ | ||
+ | \\ | ||
+ | ===== 实验十六:SDIO实验——读取SD卡信息 ===== | ||
+ | ==== 一、实验目的与意义 ==== | ||
+ | - 了解STM32 SDIO结构 | ||
+ | - 了解STM32 SDIO特征 | ||
+ | - 掌握SDIO的使用方法 | ||
+ | - 掌握STM32 HAL库中SDIO属性的配置方法 | ||
+ | - 掌握KEILMDK 集成开发环境使用方法 | ||
+ | ==== 二、实验设备及平台 ==== | ||
+ | - iCore3L 双核心板 | ||
+ | - JLINK(或相同功能)仿真器 | ||
+ | - 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线输出驱动器是开漏引脚,在此识别过程中允许并行的卡操作。对准过程以如下方式完成: | ||
+ | - 激活总线。 | ||
+ | - SDIO卡主机广播SEND_OP_COND(CMD1)以接收运行条件。 | ||
+ | - 卡以其运行条件寄存器的内容进行响应。 | ||
+ | - 不兼容的卡将被置于无效状态。 | ||
+ | - SDIO卡主机向所有有效广播ALL_SEND_CID(CMD2)。 | ||
+ | - 这些卡将发回其唯一的卡识别号(CID)并进入识别状态。 | ||
+ | - SDIO卡主机向某个地址的有效卡发出SET_RELATIVE_ADDR(CMD3)。这一新地址称为相对卡地址(RCA);它比CID更短,可对卡进行寻址。分配的卡变为待机状态。SDIO卡主机可以重新发出此命令以更改RCA。卡的RCA是最后分配的值。 | ||
+ | - SDIO卡主机对所有有效卡重复步骤5到7。 | ||
+ | === 5.SDIO接口的设备 === | ||
+ | {{ :icore3l:icore3l_arm_hal_16_1.png?direct |}} | ||
+ | === 6.SD卡物理结构 === | ||
+ | {{ :icore3l:icore3l_arm_hal_16_2.png?direct |}} | ||
+ | {{ :icore3l:icore3l_arm_hal_16_3.png?direct |}} | ||
+ | === 7.原理图 === | ||
+ | {{ :icore3l:icore3l_arm_hal_16_4.png?direct |}} | ||
+ | ==== 四、实验程序 ==== | ||
+ | 1.主函数 | ||
+ | <code c> | ||
+ | int main(void) | ||
+ | { | ||
+ | int i,j; | ||
+ | unsigned char write_buffer[512]; | ||
+ | unsigned char read_buffer[512]; | ||
+ | int error = 0; | ||
+ | HAL_Init(); | ||
+ | SystemClock_Config(); | ||
+ | MX_GPIO_Init(); | ||
+ | MX_SDIO_SD_Init(); | ||
+ | MX_USART1_UART_Init(); | ||
+ | usart1.initialize(115200); //串口波特设 | ||
+ | usart1.printf("\x0c"); //清屏 | ||
+ | usart1.printf("\033[1;32;40m"); //设置终端字体为绿色 | ||
+ | usart1.printf("Hello, I am iCore3L\r\n"); //串口信息输出 | ||
+ | if(HAL_SD_ConfigWideBusOperation(&hsd, SDIO_BUS_WIDE_4B) != HAL_OK) | ||
+ | { | ||
+ | usart1.printf("SD Card Error!\r\n"); | ||
+ | while(1) | ||
+ | { | ||
+ | LED_RED_ON; | ||
+ | HAL_Delay(500); | ||
+ | LED_RED_OFF; | ||
+ | HAL_Delay(500); | ||
+ | } | ||
+ | } | ||
+ | else | ||
+ | { | ||
+ | usart1.printf("SD Card OK!\r\n"); | ||
+ | } | ||
+ | //打印SD卡类型 | ||
+ | switch(hsd.SdCard.CardType) | ||
+ | { | ||
+ | case CARD_SDSC: | ||
+ | usart1.printf("SD CardType\t\t: CARD_SDSC\r\n"); | ||
+ | break; | ||
+ | case CARD_SDHC_SDXC: | ||
+ | usart1.printf("SD CardType\t\t: CARD_SDHC_SDXC\r\n"); | ||
+ | break; | ||
+ | case CARD_SECURED: | ||
+ | usart1.printf("SD CardType\t\t: CARD_SECURED\r\n"); | ||
+ | break; | ||
+ | } | ||
+ | usart1.printf("SD CardCapacity\t\t: %dMB\r\n",(unsigned int)(hsd.SdCard. | ||
+ | BlockNbr * hsd.SdCard.BlockSize) >> 20); //打印SD卡容量 | ||
+ | usart1.printf("SD BlockSize\t\t: %dByte\r\n",hsd.SdCard.BlockSize); | ||
+ | //打印SD卡数据块大小 | ||
+ | usart1.printf("SD RelCardAdd\t\t: %d\r\n",hsd.SdCard.RelCardAdd); | ||
+ | //打印SD卡相对地址 | ||
+ | //从缓冲区写入512字节数据到SD卡数据块 | ||
+ | for(i = 0;i < 512;i++) | ||
+ | { | ||
+ | write_buffer[i] = i % 256; | ||
+ | } | ||
+ | HAL_SD_WriteBlocks(&hsd,write_buffer,0,1,1000); | ||
+ | HAL_Delay(1); //确定写操作完成 | ||
+ | //读取SD卡数据块数据到缓冲区 | ||
+ | HAL_SD_ReadBlocks(&hsd,read_buffer,0,1,1000); | ||
+ | for(i = 0;i < 512;i ++) | ||
+ | { | ||
+ | if(write_buffer[i] != read_buffer[i]) | ||
+ | { //比较数据是否读写正确 | ||
+ | error ++; | ||
+ | } | ||
+ | } | ||
+ | if(error) | ||
+ | { //测试失败,LED闪烁 | ||
+ | usart1.printf("SD TEST FAIL!\r\n"); | ||
+ | while(1) | ||
+ | { | ||
+ | LED_RED_ON; | ||
+ | HAL_Delay(500); | ||
+ | LED_RED_OFF; | ||
+ | HAL_Delay(500); | ||
+ | } | ||
+ | } | ||
+ | else | ||
+ | { //测试成功打印读取到的数据 | ||
+ | LED_RED_ON; | ||
+ | usart1.printf("SD TEST OK!\r\n"); | ||
+ | usart1.printf("read data:\r\n"); | ||
+ | for(i = 0;i < 32;i++) | ||
+ | { | ||
+ | for(j = 0; j < 16; j ++) | ||
+ | { | ||
+ | usart1.printf(" %02X",read_buffer[i*16+j]); | ||
+ | } | ||
+ | usart1.printf("\r\n"); | ||
+ | } | ||
+ | } | ||
+ | while(1) | ||
+ | { | ||
+ | } | ||
+ | } | ||
+ | </code> | ||
+ | 2.SDIO初始化函数 | ||
+ | <code c> | ||
+ | void MX_SDIO_SD_Init(void) //SDIO初始化 | ||
+ | { | ||
+ | hsd.Instance = SDIO; | ||
+ | hsd.Init.ClockEdge = SDIO_CLOCK_EDGE_RISING; | ||
+ | hsd.Init.ClockBypass = SDIO_CLOCK_BYPASS_DISABLE; | ||
+ | hsd.Init.ClockPowerSave = SDIO_CLOCK_POWER_SAVE_DISABLE; | ||
+ | hsd.Init.BusWide = SDIO_BUS_WIDE_1B; | ||
+ | hsd.Init.HardwareFlowControl = SDIO_HARDWARE_FLOW_CONTROL_DISABLE; | ||
+ | hsd.Init.ClockDiv = 0; | ||
+ | if (HAL_SD_Init(&hsd) != HAL_OK) | ||
+ | { | ||
+ | Error_Handler(); | ||
+ | } | ||
+ | if (HAL_SD_ConfigWideBusOperation(&hsd, SDIO_BUS_WIDE_4B) != HAL_OK) | ||
+ | { | ||
+ | Error_Handler(); | ||
+ | } | ||
+ | } | ||
+ | </code> | ||
+ | 3.SD卡写数据块函数 | ||
+ | <code c> | ||
+ | HAL_SD_WriteBlocks(SD_HandleTypeDef *hsd, uint8_t *pData, uint32_t BlockAdd, uint32_t NumberOfBlocks, uint32_t Timeout) | ||
+ | //*hsd:指向SD卡的指针 | ||
+ | //*pData:指向要写入的数据的指针 | ||
+ | //BlockAdd:数据块地址 | ||
+ | //NumberOfBlocks:写入的块数 | ||
+ | //Timeout:写入超时设置 | ||
+ | </code> | ||
+ | 4.SD读数据块函数 | ||
+ | <code c> | ||
+ | HAL_SD_ReadBlocks(SD_HandleTypeDef *hsd, uint8_t *pData, uint32_t BlockAdd, uint32_t NumberOfBlocks, uint32_t Timeout) | ||
+ | //*hsd:指向SD卡的指针 | ||
+ | //*pData:指向数据读取后存放区的指针 | ||
+ | //BlockAdd:数据块地址 | ||
+ | //NumberOfBlocks:读取的块数 | ||
+ | //Timeout:读取超时设置 | ||
+ | </code> | ||
+ | 5.SDIO外设管理结构体 | ||
+ | <code c> | ||
+ | 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; | ||
+ | </code> | ||
+ | 6.SDIO 数据初始化结构体 | ||
+ | <code c> | ||
+ | typedef struct { | ||
+ | uint32_t DataTimeOut; // 数据传输超时 | ||
+ | uint32_t DataLength; // 数据长度 | ||
+ | uint32_t DataBlockSize; // 数据块大小 | ||
+ | uint32_t TransferDir; // 数据传输方向 | ||
+ | uint32_t TransferMode; // 数据传输模式 | ||
+ | uint32_t DPSM; // 数据路径状态机 | ||
+ | } SDIO_DataInitTypeDef; | ||
+ | </code> | ||
+ | 7.SD卡信息结构体 | ||
+ | <code c> | ||
+ | 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; | ||
+ | </code> | ||
+ | ==== 五、实验步骤 ==== | ||
+ | - 把仿真器与iCore3L的SWD调试口相连(直接相连或者通过转接器相连); | ||
+ | - 把iCore3L通过Micro USB线与计算机相连,为iCore3L供电; | ||
+ | - 打开PuTTY串口终端; | ||
+ | - 打开Keil MDK 开发环境,并打开本实验工程; | ||
+ | - 烧写程序到iCore3L上; | ||
+ | - 也可以进入Debug 模式,单步运行或设置断点验证程序逻辑。 | ||
+ | ==== 六、实验现象 ==== | ||
+ | 终端显示出SD卡的相关信息,并输出写入SD卡的数据。 | ||
+ | {{ :icore3l:icore3l_arm_hal_16_5.png?direct |}} | ||
+ | ===== 附录 ===== | ||
+ | 1.安装CH340驱动(双击安装,如果已安装忽略此步)\\ | ||
+ | 2.iCore3L供电后,打开计算机——属性——设备管理器——端口 | ||
+ | {{ :icore3l:icore3l_arm_hal_16_6.png?direct |}} | ||
+ | 3.打开puTTY | ||
+ | {{ :icore3l:icore3l_arm_hal_16_7.png?direct |}} | ||
+ | 4.烧写程序进行验证 |