用户工具

站点工具


icore3l_arm_16

差别

这里会显示出您选择的修订版和当前版本之间的差别。

到此差别页面的链接

后一修订版
前一修订版
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.烧写程序进行验证
icore3l_arm_16.1605665962.txt.gz · 最后更改: 2020/11/18 10:19 由 zgf