stm32 sdio调试,修改官方例程bug,已成功调通SDIO读取SD卡

这几天研究stm32操作sd卡,使用的是ST给的例程stm32_eval_sdio_sd.c,版本V4.5.0,遇到了如下问题,现一一解决。

本文由Eric Qu撰写,QQ:12430300。转载请注明出处。

代码下载请见 http://download.csdn.net/detail/cokewei/4133725 需要资源份2分,请尊重我的劳动成果,谢谢。

1:SD_WaitReadOperation()或者SD_WaitWriteOperation()函数死循环

原因:数据传输错误导致传输中断,无法满足退出等待的判断条件。

代码分析:

SD_Error SD_WaitReadOperation(void)
{
  SD_Error errorstatus = SD_OK;
 
  while ((SD_DMAEndOfTransferStatus() == RESET) && (TransferEnd == 0) && (TransferError == SD_OK))
  {}

  if (TransferError != SD_OK)
  {
    return(TransferError);
  }

  return(errorstatus);
}

代码中用了while()是导致死循环的原因。TransferEnd ,TransferError 这两个参数是在中断中修改的, SD_DMAEndOfTransferStatus() 的结束条件是DMA传输结束。

看中断例程:

SD_Error SD_ProcessIRQSrc(void)
{
  if (StopCondition == 1)
  {
    SDIO->ARG = 0x0;
    SDIO->CMD = 0x44C;
    TransferError = CmdResp1Error(SD_CMD_STOP_TRANSMISSION);
  }
  else
  {
    TransferError = SD_OK;
  }
  SDIO_ClearITPendingBit(SDIO_IT_DATAEND);
  SDIO_ITConfig(SDIO_IT_DATAEND, DISABLE);
  TransferEnd = 1;
  return(TransferError);
}

中断配置为数据传输结束中断SDIO_ITConfig(SDIO_IT_DATAEND, ENABLE);

数据传输出错导致传输中断的情况下,SD_DMAEndOfTransferStatus的判断为false,TransferEnd 和TransferError 的状态也不会改变,导致while循环无法退出。

解决:

中断配置为SDIO_ITConfig(SDIO_IT_RXOVERR|SDIO_IT_DTIMEOUT|SDIO_IT_DCRCFAIL|SDIO_IT_DATAEND, ENABLE);

中断例程修改:

SD_Error SD_ProcessIRQSrc(void)
{
 if (SDIO_GetFlagStatus(SDIO_FLAG_DTIMEOUT) != RESET)
 {
  SDIO_ClearFlag(SDIO_FLAG_DTIMEOUT);
  TransferError = SD_DATA_TIMEOUT;
 }
 else if (SDIO_GetFlagStatus(SDIO_FLAG_DCRCFAIL) != RESET)
 {
  SDIO_ClearFlag(SDIO_FLAG_DCRCFAIL);
  TransferError = SD_DATA_CRC_FAIL;
 }
 else if (SDIO_GetFlagStatus(SDIO_FLAG_RXOVERR) != RESET)
 {
  SDIO_ClearFlag(SDIO_FLAG_RXOVERR);
  TransferError = SD_RX_OVERRUN;
 }
 else if (SDIO_GetFlagStatus(SDIO_FLAG_STBITERR) != RESET)
 {
  SDIO_ClearFlag(SDIO_FLAG_STBITERR);
  TransferError = SD_START_BIT_ERR;
 }
 else
 {
  if (StopCondition == 1)
  {
  SDIO->ARG = 0x0;
  SDIO->CMD = 0x44C;
  TransferError = CmdResp1Error(SD_CMD_STOP_TRANSMISSION);
  }
  else
  {
  TransferError = SD_OK;
  }
 }
 
  SDIO_ITConfig(SDIO_IT_RXOVERR|SDIO_IT_DTIMEOUT|SDIO_IT_DCRCFAIL|SDIO_IT_DATAEND, DISABLE);
  TransferEnd = 1;
  return(TransferError);
}

这么做的作用是发生错误时也会进入中断,即使dma没有结束,也能退出while循环。

 

2:sd_init()过程失败

可能性1:按照SD规范,初始化之前需要有74个或更多个clock让sd卡同步,例程中把clock开起来后直接发送cmd0,没有同步clock,所以先修改SD_PowerON函数内SDIO_ClockCmd(ENABLE);调用之后增加200us延时。

可能性2:发生SDIO_FLAG_DCRCFAIL错误。

修改SDIO_TRANSFER_CLK_DIV 来修改数据传输速率。

按照ST例程的注释,数据传输速率不能超过25M,但是原先配置SDIO_TRANSFER_CLK_DIV=0,按照72M主频计算的话传输速度达到72/2=36M,不出错就怪了。

我现在配置SDIO_TRANSFER_CLK_DIV=2后正常。

可能性3:固件库使用不正确。使用V4.5的例程,固件库需要用V3.5.0的,试过3.2的固件库会失败。

 

3:SD_DMAEndOfTransferStatus函数内没有清标志位,按照datasheet,标志位是由手动清除的。

4:SD_ReadBlock()发生SDIO_FLAG_DCRCFAIL错误。

一开始是实践发现,先执行一下SD_ReadMultiBlocks函数,以后再执行SD_ReadBlock就正常了。很奇怪的现象吧。后来查阅了一些资料,发现SD卡的block大小并不是固定的,可以配置为512,1024等,于是怀疑是block大小配置不正确导致。检查发现SD_ReadBlock操作之前没有设置sd卡block大小,也就是cmd16,加入这个代码就像行了

  /*!< Set Block Size for Card */
  SDIO_CmdInitStructure.SDIO_Argument = (uint32_t) BlockSize;
  SDIO_CmdInitStructure.SDIO_CmdIndex = SD_CMD_SET_BLOCKLEN;
  SDIO_CmdInitStructure.SDIO_Response = SDIO_Response_Short;
  SDIO_CmdInitStructure.SDIO_Wait = SDIO_Wait_No;
  SDIO_CmdInitStructure.SDIO_CPSM = SDIO_CPSM_Enable;
  SDIO_SendCommand(&SDIO_CmdInitStructure);

  errorstatus = CmdResp1Error(SD_CMD_SET_BLOCKLEN);

  if (SD_OK != errorstatus)
  {
    return(errorstatus);
  }

其实这个操作并不是每次读操作都要执行的,如果中途不改变block大小,只要初始化的时候设置一次就可以了。

  • 14
    点赞
  • 45
    收藏
    觉得还不错? 一键收藏
  • 12
    评论
以下是基于STM32F103的SDIO程: 首先,需要在STM32CubeMX中配置SDIO和GPIO引脚。SDIO引脚需要配置为复用功能,GPIO引脚需要配置为推挽输出,具体配置如下图所示: ![SDIO配置示意图](https://i.loli.net/2021/01/27/Pc7fWqg5rL4GZ6M.png) 然后生成代码并导入到Keil或者其他开发环境中。在代码中需要引入以下头文件: ```c #include "stm32f1xx_hal.h" #include "stm32f1xx_hal_gpio.h" #include "stm32f1xx_hal_sdio.h" ``` 在main函数中初始化SDIO: ```c /* SDIO init function */ MX_SDIO_SD_Init(); ``` 其中,MX_SDIO_SD_Init()是由STM32CubeMX自动生成的SDIO初始化函数。 接下来,就可以进行SDIO信了。以下是一个简单的示程序,实现了在SD上创建一个文件,并写入一些数据: ```c #include "main.h" #include "stm32f1xx_hal.h" #include "stm32f1xx_hal_gpio.h" #include "stm32f1xx_hal_sdio.h" /* SDIO handle */ SD_HandleTypeDef hsd; /* Buffer for data */ uint8_t buffer[512]; int main(void) { /* MCU initialization */ HAL_Init(); /* System clock initialization */ SystemClock_Config(); /* SDIO init function */ MX_SDIO_SD_Init(); /* SD card initialization */ if (HAL_SD_Init(&hsd) != HAL_OK) { /* Initialization error */ Error_Handler(); } /* Enable wide bus operation */ if (HAL_SD_ConfigWideBusOperation(&hsd, SDIO_BUS_WIDE_4B) != HAL_OK) { /* Configuration error */ Error_Handler(); } /* Create a file on the SD card */ FIL file; FRESULT res = f_open(&file, "test.txt", FA_CREATE_ALWAYS | FA_WRITE); if (res != FR_OK) { /* File creation error */ Error_Handler(); } /* Write some data to the file */ for (int i = 0; i < 512; i++) { buffer[i] = i; } UINT bytes_written; res = f_write(&file, buffer, sizeof(buffer), &bytes_written); if (res != FR_OK || bytes_written != sizeof(buffer)) { /* Write error */ Error_Handler(); } /* Close the file */ f_close(&file); while (1) { /* Infinite loop */ } } /* SDIO init function */ void MX_SDIO_SD_Init(void) { /* SDIO GPIO Configuration */ GPIO_InitTypeDef GPIO_InitStruct = {0}; /* GPIO Ports Clock Enable */ __HAL_RCC_GPIOC_CLK_ENABLE(); __HAL_RCC_GPIOD_CLK_ENABLE(); __HAL_RCC_GPIOB_CLK_ENABLE(); /* Configure SDIO D0~D3, CLK and CMD pins */ GPIO_InitStruct.Pin = GPIO_PIN_8 | GPIO_PIN_9 | GPIO_PIN_10 | GPIO_PIN_11 | GPIO_PIN_12; GPIO_InitStruct.Mode = GPIO_MODE_AF_PP; GPIO_InitStruct.Pull = GPIO_NOPULL; GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_HIGH; HAL_GPIO_Init(GPIOC, &GPIO_InitStruct); GPIO_InitStruct.Pin = GPIO_PIN_2; HAL_GPIO_Init(GPIOD, &GPIO_InitStruct); GPIO_InitStruct.Pin = GPIO_PIN_15; HAL_GPIO_Init(GPIOB, &GPIO_InitStruct); /* SDIO peripheral clock enable */ __HAL_RCC_SDIO_CLK_ENABLE(); /* SDIO interrupt Init */ HAL_NVIC_SetPriority(SDIO_IRQn, 0, 0); HAL_NVIC_EnableIRQ(SDIO_IRQn); /* SDIO DMA Init */ /* DMA controller clock enable */ __HAL_RCC_DMA2_CLK_ENABLE(); /* DMA interrupt init */ /* DMA2_Channel4_5_IRQn interrupt configuration */ HAL_NVIC_SetPriority(DMA2_Channel4_5_IRQn, 0, 0); HAL_NVIC_EnableIRQ(DMA2_Channel4_5_IRQn); } ``` 需要注意的是,SDIO信需要使用FATFS文件系统库。如果还没有使用过FATFS,需要先进行FATFS的初始化和配置。
评论 12
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值