📄 sddriver.c
字号:
}
}
SD_StopMultiToken(); /* 发送数据停止令牌 send data stop token */
ret = SD_WaitBusy(SD_WAIT_WRITE); /* 等待写入的完成 wait for finishing writing */
if (ret != SD_NO_ERR)
{
SD_EndSD();
return SD_ERR_TIMEOUT_WRITE;
}
if (sds.card_type == CARDTYPE_SD)
{
ret = SD_GetNumWRBlcoks(&i); /* 读正确写入的块数 read the blocks that be written correctly */
if (ret != SD_NO_ERR)
{
SD_EndSD();
return ret;
}
if(i != blocknum)
ret = SD_ERR_WRITE_BLKNUMS; /* 正确写入块数错误 the blocks that be written correctly is error */
}
else
{
ret = SD_ReadCard_Status(2, tmp);
if (ret != SD_NO_ERR)
{
SD_EndSD();
return ret; /* 读寄存器失败 read register fail */
}
if((tmp[0] != 0) || (tmp[1] != 0))
ret = SD_ERR_WRITE_BLK; /* 响应指示写失败 response indicate write fail */
}
SD_EndSD();
return ret; /* 返回写入成功 return write sucessfully */
}
#endif
/*********************************************************************************************************************
** 函数名称: INT8U SD_EraseBlock() Name: INT8U SD_EraseBlock()
** 功能描述: 擦除SD/MMC卡中的块 Function: Erase the block of SD/MMC card
** 输 入: INT32U startaddr: 起始地址 Input: INT32U startaddr: start address
INT32U endaddr : 终止地址 INT32U endaddr : end address
** 输 出: 0: 成功 >0: 错误码 Output: 0: right >0: error code
** 注 意: startaddr 和 blocknum 建议为 sds.erase_unit 的整数倍, 因为有的卡只能以 sds.erase_unit 为单位进行擦除
*********************************************************************************************************************/
#if SD_EraseBlock_EN
INT8U SD_EraseBlock(INT32U startaddr, INT32U blocknum)
{
INT32 tmp;
INT8U ret;
SD_StartSD(); /* 向OS申请访问卡信号量 request semaphore acessed SD/MMC to OS */
if (SD_ChkCard() != 1)
{
SD_EndSD();
return SD_ERR_NO_CARD; /* 卡没完全插入卡中 card is not inserted entirely */
}
if ((startaddr + blocknum) > sds.block_num)
{
SD_EndSD();
return SD_ERR_OVER_CARDRANGE; /* 操作超出卡容量范围 operate over the card range */
}
if (SD_ChkCardWP() == 1)
{
SD_EndSD();
return SD_ERR_WRITE_PROTECT; /* 卡有写保护 */
}
tmp = blocknum - sds.erase_unit;
while(tmp >= 0) /* 每次擦除扇区 once erase is sector size */
{
ret = SD_EraseStartBlock(startaddr); /* 选择起始块地址 select start address */
if (ret != SD_NO_ERR)
{
SD_EndSD();
return ret;
}
ret = SD_EraseEndBlock(startaddr + sds.erase_unit - 1); /* 选择终止块地址 select end address */
if (ret != SD_NO_ERR)
{
SD_EndSD();
return ret;
}
ret = SD_EraseSelectedBlock(); /* 擦除所选的块 erase blocks selected */
if (ret != SD_NO_ERR)
{
SD_EndSD();
return ret;
}
startaddr += sds.erase_unit; /* 起始地址递增 */
blocknum -= sds.erase_unit;
tmp = blocknum - sds.erase_unit;
};
if (blocknum > 0) /* 擦除不够once_erase块 */
{ /* erase blocks that numbers is not enough once_erase */
ret = SD_EraseStartBlock(startaddr);
if (ret != SD_NO_ERR)
{
SD_EndSD();
return ret;
}
ret = SD_EraseEndBlock(startaddr + blocknum - 1);
if (ret != SD_NO_ERR)
{
SD_EndSD();
return ret;
}
ret = SD_EraseSelectedBlock();
if (ret != SD_NO_ERR)
{
SD_EndSD();
return ret;
}
}
SD_EndSD();
return SD_NO_ERR; /* 返回擦除成功 return erase sucessfully */
}
#endif
/*****************************************************************
下面为子程序
*****************************************************************/
/*******************************************************************************************************************
** 函数名称: INT8U SD_GetCardInfo() Name: INT8U SD_GetCardInfo()
** 功能描述: 获得SD/MMC卡的信息 Function: get the information of SD/MMC card
** 输 入: INT8U cardtype: 卡类型 Input: INT8U cardtype: card type
** 输 出: 0: 成功 >0: 错误码 Output: 0: right >0: error code
*******************************************************************************************************************/
INT8U SD_GetCardInfo()
{
INT32U tmp;
INT8U csdbuf[16],ret;
ret = SD_ReadCSD(16, csdbuf); /* 读CSD寄存器 read CSD register */
if (ret != SD_NO_ERR)
return ret;
SD_CalTimeout(csdbuf); /* 计算超时时间值 calculate timeout value */
/* 计算块的最大长度 */ /* calculate the size of a sector */
sds.block_len = 1 << (csdbuf[READ_BL_LEN_POS] & READ_BL_LEN_MSK); /* (2 ^ READ_BL_LEN) */
/* 计算卡中块的个数 */ /* calculate the sector numbers of the SD Card */
sds.block_num = ((csdbuf[C_SIZE_POS1] & C_SIZE_MSK1) << 10) +
(csdbuf[C_SIZE_POS2] << 2) +
((csdbuf[C_SIZE_POS3] & C_SIZE_MSK3) >> 6) + 1; /* (C_SIZE + 1)*/
tmp = ((csdbuf[C_SIZE_MULT_POS1] & C_SIZE_MULT_MSK1) << 1) +
((csdbuf[C_SIZE_MULT_POS2] & C_SIZE_MULT_MSK2) >> 7) + 2; /* (C_SIZE_MULT + 2) */
/* 获得卡中块的数量 */ /* get the block numbers in card */
sds.block_num = sds.block_num * (1 << tmp); /* (C_SIZE + 1) * 2 ^ (C_SIZE_MULT + 2) */
/* 计算擦除的单位(单位: 块) */
if (sds.card_type == CARDTYPE_MMC)
{
tmp = ((csdbuf[ERASE_GRP_SIZE_POS] & ERASE_GRP_SIZE_MSK) >> 2) + 1; /* (ERASE_GRP_SIZE + 1) */
/* (ERASE_GRP_SIZE + 1) * (ERASE_GRP_MULTI + 1) */
tmp *= ((csdbuf[ERASE_GRP_MULTI_POS1] & ERASE_GRP_MULTI_MSK1) << 3) +
((csdbuf[ERASE_GRP_MULTI_POS2] & ERASE_GRP_MULTI_MSK2) >> 5) + 1;
}
else /*calculate the size of sector */
tmp = ((csdbuf[SECTOR_SIZE_POS1] & SECTOR_SIZE_MSK1) << 1) +
((csdbuf[SECTOR_SIZE_POS2] & SECTOR_SIZE_MSK2) >> 7) + 1; /* SD: SECTOR_SIZE */
sds.erase_unit = tmp; /* 擦除单位(块) */
return SD_NO_ERR; /* 返回执行成功 return perform sucessfully */
}
/*******************************************************************************************************************
** 函数名称: INT8U SD_CalTimeout() Name: INT8U SD_CalTimeout()
** 功能描述: 计算读/写/擦超时时间 Function: calculate timeout of reading,writing,erasing
** 输 入: INT8U *csdbuf : CSD寄存器内容 Input: INT8U *csdbuf : CSD register content
** 输 出: 0: 成功 >0: 错误码 Output: 0: right >0: error code
*******************************************************************************************************************/
INT8U SD_CalTimeout(INT8U *csdbuf)
{
INT32U tmp;
INT8U time_u,time_v,fator;
sds.timeout_read = READ_TIMEOUT_100MS; /* 默认读超时为100ms */
sds.timeout_write = WRITE_TIMEOUT_250MS; /* 默认写超时为250ms */
sds.timeout_erase = WRITE_TIMEOUT_250MS;
time_u = (csdbuf[TAAC_POS] & TAAC_MSK); /* 读超时时间单位 read timeout unit */
time_v = (csdbuf[TAAC_POS] & NSAC_MSK) >> 3; /* 读超时时间值 read timeout value */
fator = (csdbuf[R2WFACTOR_POS] & R2WFACTOR_MSK) >> 2; /* 读超时时间因数 read timeout factor */
if(time_v == 0) return SD_ERR_CARD_PARAM; /* 卡参数有错误 card parameter is error */
tmp = SPI_CLOCK * time_value[time_v] / 10 / time_unit[time_u]; /* TACC * f (单位 unit: clock) */
tmp = tmp + csdbuf[NSAC_POS] * 100; /* TACC * f + NSAC * 100 (单位 unit: clock) */
/* 计算得到的超时值 the timeout value of being calculated */
sds.timeout_read = tmp;
sds.timeout_write = tmp * r2w_fator[fator]; /* (TACC * f + NSAC * 100) * R2WFACTOR (单位 unit:clock)*/
if (sds.card_type == CARDTYPE_SD)
{
sds.timeout_read = sds.timeout_read * 100 / 8; /* 实际值为计算值的100倍 */
sds.timeout_write = sds.timeout_write * 100 / 8;
if (sds.timeout_read > READ_TIMEOUT_100MS) /* 取计算值与默认值中的最小值 */
sds.timeout_read = READ_TIMEOUT_100MS;
if (sds.timeout_write > WRITE_TIMEOUT_250MS)
sds.timeout_write = WRITE_TIMEOUT_250MS;
}
else
{
sds.timeout_read = sds.timeout_read * 10 / 8; /* 实际值为计算值的10倍 */
sds.timeout_write = sds.timeout_write * 10 / 8;
}
sds.timeout_erase = sds.timeout_write;
#if SD_UCOSII_EN
sds.timeout_read = (sds.timeout_read << 3) * OS_TICKS_PER_SEC / SPI_CLOCK;
sds.timeout_write =(sds.timeout_write << 3) * OS_TICKS_PER_SEC / SPI_CLOCK;
sds.timeout_erase = sds.timeout_write; /* (单位 unit: os tick) */
#endif
return SD_NO_ERR;
}
/*******************************************************************************************************************
** 函数名称: INT8U SD_ActiveInit() Name: INT8U SD_ActiveInit()
** 功能描述: 激活卡,并获得卡型 Function: active card, and get the card type
** 输 入: 无 Input: NULL
** 输 出: 0: 成功 >0: 错误码 Output: 0: right >0: error code
** 函数说明: 该命令不断重复发送到SD卡,直到响应R1的Bit0(Idle)位为0,表示SD卡内部初始化处理完成。
当响应的Idle位为0时,SD卡就完全进入SPI模式了。当然重复发送命令CMD1是有次数限制的,
最大次数为宏定义SD_IDLE_WAIT_MAX.
*******************************************************************************************************************/
INT8U SD_ActiveInit(void)
{
INT8U param[4] = {0,0,0,0},resp[5],ret;
INT32U i = 0;
do
{ /* 发出CMD1, 查询卡的状态, send CMD1 to poll card status */
ret = SD_SendCmd(CMD1, param, CMD1_R, resp);
if (ret != SD_NO_ERR)
return ret;
i ++;
}while (((resp[0] & MSK_IDLE) == MSK_IDLE) && (i <= SD_IDLE_WAIT_MAX));
/* 如果响应R1的最低位Idle位为1,则继续循环 */
/* if response R1 Idle bit is 1,continue recycle */
if (i >= SD_IDLE_WAIT_MAX)
return SD_ERR_TIMEOUT_WAITIDLE; /* 超时,返回错误 time out,return error */
ret = SD_SendCmd(CMD55, param, CMD55_R, resp);
if (ret != SD_NO_ERR)
return ret;
ret = SD_SendCmd(ACMD41, param, ACMD41_R, resp); /* 激活内部初始化命令 active card to initialize process internal */
if (ret != SD_NO_ERR)
return SD_ERR_UNKNOWN_CARD;
if ((resp[0] & 0xFE) == 0)
sds.card_type = CARDTYPE_SD; /* 是SD卡 the card is SD card */
else
sds.card_type = CARDTYPE_MMC; /* 是MMC卡 the card is MMC card */
return SD_NO_ERR;
}
/*******************************************************************************************************************
** 函数名称: void SD_StartSD() Name: INT8U SD_StartSD()
** 功能描述: 向操作系统申请访问SD卡的权限 Function: request the right of operating sd to OS
** 输 入: 无 Input: NULL
** 返 回: 无 return: NULL
********************************************************************************************************************/
void SD_StartSD(void)
{
#if SD_UCOSII_EN
INT8U ret;
OSSemPend(pSemSD, 0, &ret); /* 等待访问卡信号量可用 wait for semaphore that accessed Card */
#endif
}
/*******************************************************************************************************************
** 函数名称: void SD_EndSD() Name: INT8U SD_EndSD()
** 功能描述: 访问SD卡的权限归还操作系统 Function: return the right of operating sd to OS
** 输 入: 无 Input: NULL
** 返 回: 无 return: NULL
********************************************************************************************************************/
void SD_EndSD(void)
{
#if SD_UCOSII_EN
OSSemPost(pSemSD); /* 将访问卡信号量还给操作系统 return the semaphore accessing Card to OS */
#endif
}
/*******************************************************************************************************************
** 函数名称: INT16U SD_GetZLGSDVer() Name: INT16U SD_GetZLGSDVer()
** 功能描述: 得到ZLG/SD的版本号 Function: get the version of ZLG/SD
** 输 入: 无 Input: NULL
** 输 出: 版本号 Output: Version
** 版本记录:
V1.0 支持访问SD 卡,只能在前后台系统运行
V2.0 支持访问SD和MMC卡,可运行于前后系统或UCOS-II
*******************************************************************************************************************/
INT16U SD_GetZLGSDVer(void)
{
return 0x0200; /* 版本号为 2.00 */
}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -