SD卡扇区擦除之整卡擦除(以及一些细节问题和疑惑)

前言
  1. 其实这个博文是强迫症的产物,对产物;如果要在一个SD卡上建立文件系统FATFS,我带从这张卡的第0个字节,清除到最后一个字节,确保卡上“无残留”,从而不会影响我后续的工作;当整张卡被清除的那一刻,开心!!!
  2. 文章中的例程包含标准容量卡SDSC的和高容量卡SDHC的例程;
  3. 以我16G的金士顿SD卡为例;
流程
  • 首先你需要获取SD卡上的CSD寄存器值,从而计算出当前卡的准确容量,可以看我下面这个博客:https://blog.youkuaiyun.com/wuyuzun/article/details/90581825
    这里为啥说当前卡的准确容量,来看下面这个图片比较:
    计算机显示我的内存卡的总容量如下
    在这里插入图片描述
    通过STM32的 SD_GetCardStatus() 函数获取SD卡容量,计算过程看上面我那个博客就行。通过串口打印出来如下:
    在这里插入图片描述
    显然:15707668480 != 15690891264;但是两个差值= 16777216 = 16KB;难道这是主引导区MBR?这就是我异或得地方,这是怎么回事?如果有知道的朋友,还麻烦给告诉一下,不胜感激;
    所以这里我们以大的15707668480值为此16GSD卡的字节内存空间;

  • 执行命令
    图片来源于SD卡V2.0使用手册在这里插入图片描述

  1. SDSC卡 (0~2G,包含2G)是以字节位单位擦除;
    ERASE_WR_BLK_START和ERASE_WR_BLK_END是以字节位单位的物理地址
    地址范围:[ERASE_WR_BLK_START,ERASE_WR_BLK_END] == [0,0x001FFFFF]

  2. SDHC卡(2-32G,包括32G)以块为单位进行擦除(固定为512字节);
    ERASE_WR_BLK_START和ERASE_WR_BLK_END是以字节位单位的块物理地址
    地址范围: ERASE_WR_BLK_START,ERASE_WR_BLK_END] == [0,0x01FF FFFF];(16G);
    4G: [0,0xFFFF FFFF]
    8G: [0,0x00FF FFFF‬]
    32G: [0,0x03FF FFFF]
    当然上面这都是理论值;实际值是以 15707668480字节为基准的 [0,0x01D41FFF],这里的这个实际也是本文讨论的重点;

  • 相关代码如下

代码是移植于STM32的SD卡例程

/**
  * @brief  Allows to erase memory Block area specified for the given card.
  * @param  Block_startaddr: Sets the address of the first write block to be erased.
  * @param  Block_endaddr: Sets the address of the last write block of the continuous range to be erased.
  * @retval SD_Error: SD Card Error code.
  *注意:参数里的起始地址和中止地址都是以块地址为单位;可以擦除整个16G空间;
		这里有几个值一定要特别注意:
		0x01D42000是我的16G的SD卡,以512字节位块单位,状态寄存器返回的内存空间大小,但是这个值并不能作为擦除空间的endaddr;
		0x01D3E000是电脑显示的16GSD卡的以512字节位块单位的地址空间,可以作为endaddr;
		0x0x01D41FFF才是能作为最终endaddr的值;
  */
SD_Error SD_Block_Erase(uint32_t Block_startaddr, uint32_t Block_endaddr)
{
  SD_Error errorstatus = SD_OK;
  uint32_t delay = 0;
  __IO uint32_t maxdelay = 0;
  uint8_t cardstate = 0;

  /*!< Check if the card coomnd class supports erase command */
  if (((CSD_Tab[1] >> 20) & SD_CCCC_ERASE) == 0)   //0x00000020
  {
    errorstatus = SD_REQUEST_NOT_APPLICABLE;
    return(errorstatus);
  }

  maxdelay = 120000 / ((SDIO->CLKCR & 0xFF) + 2);

  if (SDIO_GetResponse(SDIO_RESP1) & SD_CARD_LOCKED)
  {
    errorstatus = SD_LOCK_UNLOCK_FAILED;
    return(errorstatus);
  }
	
  /*!< According to sd-card spec 1.0 ERASE_GROUP_START (CMD32) and erase_group_end(CMD33) */
  if ((SDIO_STD_CAPACITY_SD_CARD_V1_1 == CardType) || (SDIO_STD_CAPACITY_SD_CARD_V2_0 == CardType) || (SDIO_HIGH_CAPACITY_SD_CARD == CardType))
  {
    /*!< Send CMD32 SD_ERASE_GRP_START with argument as addr  */
    SDIO_CmdInitStructure.SDIO_Argument = Block_startaddr;
    SDIO_CmdInitStructure.SDIO_CmdIndex = SD_CMD_SD_ERASE_GRP_START;
    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_SD_ERASE_GRP_START);
    if (errorstatus != SD_OK)
    {
      return(errorstatus);
    }

    /*!< Send CMD33 SD_ERASE_GRP_END with argument as addr  */
    SDIO_CmdInitStructure.SDIO_Argument = Block_endaddr;
    SDIO_CmdInitStructure.SDIO_CmdIndex = SD_CMD_SD_ERASE_GRP_END;
    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_SD_ERASE_GRP_END);
    if (errorstatus != SD_OK)
    {
      return(errorstatus);
    }
  }

  /*!< Send CMD38 ERASE */
  SDIO_CmdInitStructure.SDIO_Argument = 0;
  SDIO_CmdInitStructure.SDIO_CmdIndex = SD_CMD_ERASE;
  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_ERASE);

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

  for (delay = 0; delay < maxdelay; delay++)
  {}

  /*!< Wait till the card is in programming state */
  errorstatus = IsCardProgramming(&cardstate);

  while ((errorstatus == SD_OK) && ((SD_CARD_PROGRAMMING == cardstate) || (SD_CARD_RECEIVING == cardstate)))
  {
    errorstatus = IsCardProgramming(&cardstate);
  }

  return(errorstatus);
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值