?? sddriver.c
字號:
INT8U SdSpi_ActiveInit(sd_struct *sds);
/*******************************************************************************************************************
** 函數名稱: SD_Initialize()
**
** 功能描述: SD/MMC 卡初始化: 復位卡,識別卡,獲取卡相關信息,使卡進入數據傳輸模式
**
** 輸 入: sd_struct *sds: SD/MMC卡信息結構體
**
** 輸 出: 無
**
** 返 回 值: 0: 正確 >0: 錯誤碼, 見 sddriver.h 文件
********************************************************************************************************************/
INT8U SD_Initialize(sd_struct *sds)
{
INT32U ocr;
INT8U ret;
#if SD_UCOSII_EN
if (sds->pSemSD == NULL)
{
sds->pSemSD = OSSemCreate(1); /* 創建訪問SD/MMC卡信號量 create SD/MMC semaphore */
if (sds->pSemSD == NULL)
return SD_ERR_CREATE_SEMSD;
}
#endif
SD_RequestOSSem(sds);
SdSpiHal_Initialize(sds); /* 初始化讀寫SD卡的硬件條件 Initialize the hardware that access SD Card */
if (!SdHal_CheckCard(sds)) /* 檢查卡是否插入 check weather card is inserted */
{
ret = SD_ERR_NO_CARD;
goto SD_ERR;
}
SdSpiHal_CSAssert(sds); /* 1. 置CS為低 assert CS */
SdSpi_Delay(sds, 25); /* 2. 至少延時 74 clock delay more than 74 clock */
SdSpiHal_CSDeassert(sds); /* 3. 置CS為高 dessert CS */
SdSpi_Delay(sds, 2); /* 4. 延時2(8 clock) delay 2(8 clock) */
ret = SpiCmd_Go_Idle_State(sds); /* 5. 發出CMDO命令復位SD卡 send CMD0 command to reset sd card */
if (ret != SD_NO_ERR)
goto SD_ERR;
ret = SdSpi_ActiveInit(sds); /* 6. 激活卡進入初始化過程. active card initialize process */
if (ret != SD_NO_ERR)
goto SD_ERR;
ret = SpiCmd_ReadOCR(sds, &ocr); /* 7. 讀OCR寄存器,查詢卡支持的電壓值 read OCR register,get the supported voltage */
if (ret != SD_NO_ERR)
goto SD_ERR;
if ((ocr & MSK_OCR_33) != MSK_OCR_33)
{
ret = SD_ERR_VOL_NOTSUSP; /* 不支持3.3V,返回錯誤碼 not support 3.3V,return error code */
goto SD_ERR;
}
SdSpiHal_SetMCIClock(sds, SD_RATE_NORMAL); /* 8. 設置訪問卡的clock為標準clock */
#if SD_CRC_EN
ret = SpiCmd_CRC_ON_OFF(sds, 1); /* 使能CRC校驗 enable CRC check */
#else
ret = SpiCmd_CRC_ON_OFF(sds, 0); /* 禁止CRC校驗 disable CRC check */
#endif
if (ret != SD_NO_ERR)
goto SD_ERR;
ret = SpiCmd_Set_BlockLen(sds, SD_BLOCKSIZE); /* 9. 設置塊的長度: 512Bytes Set the block length: 512Bytes */
if (ret != SD_NO_ERR)
goto SD_ERR;
ret = SDMMC_GetCardInfo(sds); /* 10. 讀CSD寄存器,獲取SD卡信息 read CSD register, get the information of SD card */
if (ret != SD_NO_ERR)
goto SD_ERR;
ret = SDMMC_CalTimeout(sds); /* 計算超時時間值 */
if (ret != SD_NO_ERR)
goto SD_ERR;
SD_ReleaseOSSem(sds);
return SD_NO_ERR; /* 初始化成功 initialize sucessfully */
SD_ERR:
SD_ReleaseOSSem(sds);
return ret;
}
/*******************************************************************************************************************
** 函數名稱: SD_ReadBlock()
**
** 功能描述: SPI模式下, 從SD/MMC卡中讀出一個數據塊
**
** 輸 入: sd_struct *sds : SD/MMC卡信息結構體
** INT32U blockaddr: 以塊為單位的塊地址, 例如, 卡開始的0 ~ 511字節為塊地址0, 512 ~ 1023字節的塊地址為1
**
** 輸 出: INT8U *recbuf : 接收緩沖區,長度固定為 512 字節
**
** 返 回 值: 0: 正確 >0: 錯誤碼, 見 sddriver.h 文件
********************************************************************************************************************/
INT8U SD_ReadBlock(sd_struct *sds, INT32U blockaddr, INT8U *recbuf)
{
INT8U ret;
SD_RequestOSSem(sds); /* 向OS申請訪問卡信號量 request semaphore acessed SD/MMC to OS */
if (!SdHal_CheckCard(sds))
{
SD_ReleaseOSSem(sds);
return SD_ERR_NO_CARD; /* 卡沒完全插入卡中 card is not inserted entirely */
}
if (blockaddr > sds->block_num)
{
SD_ReleaseOSSem(sds);
return SD_ERR_OVER_CARDRANGE; /* 操作超出卡容量范圍 operate over the card range */
}
ret = SpiCmd_Read_Single_Block(sds, blockaddr); /* 讀單塊命令 read single blocks command */
if (ret != SD_NO_ERR)
{
SD_ReleaseOSSem(sds);
return ret;
}
ret = SdSpi_ReadBlockData(sds, SD_BLOCKSIZE, recbuf); /* 讀出數據 read data from sd card */
SD_ReleaseOSSem(sds); /* 歸還訪問卡信號量 return semaphore acessed SD/MMC to OS */
return ret;
}
/*******************************************************************************************************************
** 函數名稱: SD_ReadMultiBlock()
**
** 功能描述: SPI模式下, 從SD/MMC卡中讀出多個數據塊
**
** 輸 入: sd_struct *sds : SD/MMC卡信息結構體
** INT32U blockaddr: 以塊為單位的塊地址
** INT32U blocknum : 要讀取的塊的個數
**
** 輸 出: INT8U *recbuf : 接收緩沖區, 長度512 * blocknum 字節
**
** 返 回 值: 0: 正確 >0: 錯誤碼, 見 sddriver.h 文件
********************************************************************************************************************/
#if SD_ReadMultiBlock_EN
INT8U SD_ReadMultiBlock(sd_struct *sds, INT32U blockaddr, INT32U blocknum, INT8U *recbuf)
{
INT32U i;
INT8U ret;
SD_RequestOSSem(sds); /* 向OS申請訪問卡信號量 request semaphore acessed SD/MMC to OS */
if (!SdHal_CheckCard(sds))
{
SD_ReleaseOSSem(sds);
return SD_ERR_NO_CARD; /* 卡沒完全插入卡中 card is not inserted entirely */
}
if ((blockaddr + blocknum) > sds->block_num)
{
SD_ReleaseOSSem(sds);
return SD_ERR_OVER_CARDRANGE; /* 操作超出卡容量范圍 operate over the card range */
}
ret = SpiCmd_Read_Multiple_Block(sds, blockaddr); /* 讀多塊命令 read multiple blocks command */
if (ret != SD_NO_ERR)
{
SD_ReleaseOSSem(sds); /* 歸還訪問卡信號量 return semaphore acessed SD/MMC to OS */
return ret;
}
for (i = 0; i < blocknum; i++)
{ /* 讀出數據 read data from SD/MMC card */
ret = SdSpi_ReadBlockData(sds, SD_BLOCKSIZE, recbuf);
if (ret == SD_NO_ERR)
recbuf = recbuf + SD_BLOCKSIZE;
else
{
SD_ReleaseOSSem(sds);
return ret;
}
}
ret = SpiCmd_Stop_Transmission(sds); /* 結束數據傳輸 stop transmission operation */
SD_ReleaseOSSem(sds);
return ret;
}
#endif
/*******************************************************************************************************************
** 函數名稱: SD_WriteBlock()
**
** 功能描述: SPI模式下, 向SD/MMC卡中寫入一個塊
**
** 輸 入: sd_struct *sds : SD/MMC卡信息結構體
** INT32U blockaddr: 以塊為單位的塊地址, 例如, 卡開始的0 ~ 511字節為塊地址0, 512 ~ 1023字節的塊地址為1
** INT8U *sendbuf : 發送緩沖區,長度固定為 512 字節
**
** 輸 出: 無
**
** 返 回 值: 0: 正確 >0: 錯誤碼, 見 sddriver.h 文件
********************************************************************************************************************/
INT8U SD_WriteBlock(sd_struct *sds, INT32U blockaddr, INT8U *sendbuf)
{
INT8U ret,tmp[2];
SD_RequestOSSem(sds); /* 向OS申請訪問卡信號量 request semaphore acessed SD/MMC to OS */
if (!SdHal_CheckCard(sds))
{
SD_ReleaseOSSem(sds);
return SD_ERR_NO_CARD; /* 卡沒完全插入卡中 card is not inserted entirely */
}
if (blockaddr > sds->block_num)
{
SD_ReleaseOSSem(sds);
return SD_ERR_OVER_CARDRANGE; /* 操作超出卡容量范圍 operate over the card range */
}
if (SdHal_CheckCardWP(sds))
{
SD_ReleaseOSSem(sds);
return SD_ERR_WRITE_PROTECT; /* 卡有寫保護 */
}
ret = SpiCmd_Write_Single_Block(sds, blockaddr); /* 寫單塊命令 write single block */
if (ret != SD_NO_ERR)
{
SD_ReleaseOSSem(sds);
return ret;
}
ret = SdSpi_WriteBlockData(sds, 0, SD_BLOCKSIZE, sendbuf); /* 寫入數據 write data */
if (ret == SD_NO_ERR) /* 讀Card Status寄存器, 檢查寫入是否成功 */
{ /* read Card Status register to check write wheather sucessfully */
ret = SpiCmd_Send_Status(sds, 2, tmp);
if (ret != SD_NO_ERR)
{
SD_ReleaseOSSem(sds);
return ret; /* 讀寄存器失敗 read register fail */
}
if((tmp[0] != 0) || (tmp[1] != 0))
{
SD_ReleaseOSSem(sds);
ret = SD_ERR_WRITE_BLK; /* 響應指示寫失敗 response indicate write fail */
}
}
SD_ReleaseOSSem(sds);
return ret; /* 返回寫入結果 return the result of writing */
}
/*******************************************************************************************************************
** 函數名稱: SD_WriteMultiBlock()
**
** 功能描述: SPI模式下, 向SD/MMC卡中寫入多個數據塊
**
** 輸 入: sd_struct *sds : SD/MMC卡信息結構體
** INT32U blockaddr: 以塊為單位的塊地址
** INT32U blocknum : 要寫入的塊的個數
** INT8U *sendbuf : 發送緩沖區, 長度 512 * blocknum 字節
**
** 輸 出: 無
**
** 返 回 值: 0: 正確 >0: 錯誤碼, 見 sddriver.h 文件
********************************************************************************************************************/
#if SD_WriteMultiBlock_EN
INT8U SD_WriteMultiBlock(sd_struct *sds, INT32U blockaddr, INT32U blocknum, INT8U *sendbuf)
{
INT32U i;
INT8U ret,tmp[2];
SD_RequestOSSem(sds); /* 向OS申請訪問卡信號量 request semaphore acessed SD/MMC to OS */
if (!SdHal_CheckCard(sds))
{
SD_ReleaseOSSem(sds);
return SD_ERR_NO_CARD; /* 卡沒完全插入卡中 card is not inserted entirely */
}
if ((blockaddr + blocknum) > sds->block_num)
{
SD_ReleaseOSSem(sds);
return SD_ERR_OVER_CARDRANGE; /* 操作超出卡容量范圍 operate over the card range */
}
if (SdHal_CheckCardWP(sds))
{
SD_ReleaseOSSem(sds);
return SD_ERR_WRITE_PROTECT; /* 卡有寫保護 */
}
ret = SpiCmd_Write_Multiple_Block(sds, blockaddr); /* 寫多塊命令 write multiple blocks command */
if (ret != SD_NO_ERR)
{
SD_ReleaseOSSem(sds);
return ret;
}
for (i = 0; i < blocknum; i++)
{
ret = SdSpi_WriteBlockData(sds, 1, SD_BLOCKSIZE, sendbuf); /* 寫入數據 write data */
if (ret == SD_NO_ERR)
sendbuf = sendbuf + SD_BLOCKSIZE;
else
{ /* 寫失敗 write fail */
SpiCmd_Stop_Transmission(sds); /* 停止數據傳輸 stop data transmission */
SdSpi_WaitBusy(sds, SD_WAIT_WRITE); /* 等待 waiting */
SD_ReleaseOSSem(sds);
return ret;
}
}
SdSpi_StopMultiToken(sds); /* 發送數據停止令牌 send data stop token */
ret = SdSpi_WaitBusy(sds, SD_WAIT_WRITE); /* 等待寫入的完成 wait for finishing writing */
if (ret != SD_NO_ERR)
{
SD_ReleaseOSSem(sds);
return SD_ERR_TIMEOUT_WRITE;
}
if (sds->card_type == CARDTYPE_SD)
{
ret = SpiCmd_Send_Num_WR_Blcoks(sds, &i); /* 讀正確寫入的塊數 read the blocks that be written correctly */
if (ret != SD_NO_ERR)
{
SD_ReleaseOSSem(sds);
return ret;
}
if(i != blocknum)
ret = SD_ERR_WRITE_BLKNUMS; /* 正確寫入塊數錯誤 the blocks that be written correctly is error */
}
else
{
ret = SpiCmd_Send_Status(sds, 2, tmp);
if (ret != SD_NO_ERR)
{
SD_ReleaseOSSem(sds);
return ret; /* 讀寄存器失敗 read register fail */
}
if((tmp[0] != 0) || (tmp[1] != 0))
ret = SD_ERR_WRITE_BLK; /* 響應指示寫失敗 response indicate write fail */
}
SD_ReleaseOSSem(sds);
return ret; /* 返回寫入成功 return write sucessfully */
}
#endif
/*****************************************************************
下面為子程序
*****************************************************************/
/*******************************************************************************************************************
** 函數名稱: SdSpi_ActiveInit()
**
** 功能描述: SPI模式下, 激活卡,并獲得卡型
**
** 輸 入: sd_struct *sds : SD/MMC卡信息結構體
**
** 輸 出: 無
**
** 返 回 值: 0: 正確 >0: 錯誤碼, 見 sddriver.h 文件
**
** 函數說明: SpiCmd_Send_OP_Cond()不斷重復發送CMD0到卡,直到響應R1的Bit0(Idle)位為0,表示卡內部初始化處理完成。
當響應的Idle位為0時,SD卡就完全進入SPI模式了。當然重復發送命令CMD1是有次數限制的,
最大次數為宏定義SD_IDLE_WAIT_MAX.
********************************************************************************************************************/
INT8U SdSpi_ActiveInit(sd_struct *sds)
{
INT8U param[4] = {0,0,0,0},resp[5],ret;
ret = SpiCmd_Send_OP_Cond(sds); /* 激活卡進入初始化過程, 使卡退出Idle狀態 */
if (ret != SD_NO_ERR)
return ret;
ret = SdSpi_SendCmd(sds, CMD55, param, CMD55_R, resp);
if (ret != SD_NO_ERR)
return ret;
ret = SdSpi_SendCmd(sds, 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;
}
#endif //end of #if (!SDBUS_MODE_EN)
?? 快捷鍵說明
復制代碼
Ctrl + C
搜索代碼
Ctrl + F
全屏模式
F11
切換主題
Ctrl + Shift + D
顯示快捷鍵
?
增大字號
Ctrl + =
減小字號
Ctrl + -