使用原子哥的SDIO SD卡程序,发现上电的时候长时间进入了死循环

2019-07-21 06:23发布

本帖最后由 ZDawn 于 2018-1-9 14:31 编辑

    用别的板子烧了原子哥的SDIO程序之后,在 “查找SD卡的SCR寄存器值” 的函数子函数里面会死循环,FIFO无接收数据,导致SDIO->STA值一直为零。但是这种情况仅限于普通的上电。
    我用keil 和JLINK在线调试的话第一次也会进入死循环,但是用keil 调试的复位按钮之后,再重新执行就能正常运行,不再进入那个死循环。
    正常上电的话偶尔等1分钟也能读到FIFO的值从而跳出死循环。
    请问这会是什么原因导致每次的第一次上电时候,SDIO->STA值一直为零??(查询模式和DMA模式都是一样的结果)


    以下是程序:


//查找SD卡的SCR寄存器值
//rca:卡相对地址
//pscr:数据缓存区(存储SCR内容)
//返回值:错误状态                  
SD_Error FindSCR(u16 rca,u32 *pscr)
{
         u8 ii=50;
        
        u32 index = 0;
        SD_Error errorstatus = SD_OK;
        u32 tempscr[2]={0,0};  
         
        
        SDIO_CmdInitStructure.SDIO_Argument = (uint32_t)8;         //发送CMD16,短响应,设置Block Size为8字节        
        SDIO_CmdInitStructure.SDIO_CmdIndex = SD_CMD_SET_BLOCKLEN; //         cmd16
        SDIO_CmdInitStructure.SDIO_Response = SDIO_Response_Short;  //r1
        SDIO_CmdInitStructure.SDIO_Wait = SDIO_Wait_No;
        SDIO_CmdInitStructure.SDIO_CPSM = SDIO_CPSM_Enable;
        SDIO_SendCommand(&SDIO_CmdInitStructure);
        delay_ms(1);
        errorstatus=CmdResp1Error(SD_CMD_SET_BLOCKLEN);

        if(errorstatus!=SD_OK)return errorstatus;            

        SDIO_CmdInitStructure.SDIO_Argument = (uint32_t) RCA << 16;
        SDIO_CmdInitStructure.SDIO_CmdIndex = SD_CMD_APP_CMD;//发送CMD55,短响应         
        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_APP_CMD);
        
         if(errorstatus!=SD_OK)return errorstatus;

        SDIO_DataInitStructure.SDIO_DataTimeOut = SD_DATATIMEOUT;
        SDIO_DataInitStructure.SDIO_DataLength = 8;  //8个字节长度,block为8字节,SD卡到SDIO.
        SDIO_DataInitStructure.SDIO_DataBlockSize = SDIO_DataBlockSize_8b  ;  //块大小8byte
        SDIO_DataInitStructure.SDIO_TransferDir = SDIO_TransferDir_ToSDIO;
        SDIO_DataInitStructure.SDIO_TransferMode = SDIO_TransferMode_Block;
        SDIO_DataInitStructure.SDIO_DPSM = SDIO_DPSM_Enable;
        SDIO_DataConfig(&SDIO_DataInitStructure);               
        
//        delay_ms(1);
        
        SDIO_CmdInitStructure.SDIO_Argument = 0x0;
        SDIO_CmdInitStructure.SDIO_CmdIndex = SD_CMD_SD_APP_SEND_SCR;        //发送ACMD51,短响应,参数为0        
        SDIO_CmdInitStructure.SDIO_Response = SDIO_Response_Short;  //r1
        SDIO_CmdInitStructure.SDIO_Wait = SDIO_Wait_No;
        SDIO_CmdInitStructure.SDIO_CPSM = SDIO_CPSM_Enable;
        SDIO_SendCommand(&SDIO_CmdInitStructure);
//        delay_ms(1);
        
         errorstatus=CmdResp1Error(SD_CMD_SD_APP_SEND_SCR);
        
         if(errorstatus!=SD_OK)return errorstatus;
        
//        delay_ms(1);
        
       //进入死循环的段落,if 语句一直不通过判定
         while(!(SDIO->STA&(SDIO_FLAG_RXOVERR|SDIO_FLAG_DCRCFAIL|SDIO_FLAG_DTIMEOUT|SDIO_FLAG_DBCKEND|SDIO_FLAG_STBITERR)))
        {
                if(SDIO_GetFlagStatus(SDIO_FLAG_RXDAVL) != RESET)//接收FIFO数据可用
                {
                        *(tempscr+index)=SDIO_ReadData();        //读取FIFO内容
                        index++;
                        if(index>=2)break;
                }
        
        }
        if(SDIO_GetFlagStatus(SDIO_FLAG_DTIMEOUT) != RESET)                //数据超时错误
                {                                                                                   
                         SDIO_ClearFlag(SDIO_FLAG_DTIMEOUT);         //清错误标志
                        return SD_DATA_TIMEOUT;
                 }else if(SDIO_GetFlagStatus(SDIO_FLAG_DCRCFAIL) != RESET)        //数据块CRC错误
                {
                         SDIO_ClearFlag(SDIO_FLAG_DCRCFAIL);                  //清错误标志
                        return SD_DATA_CRC_FAIL;                  
                }else if(SDIO_GetFlagStatus(SDIO_FLAG_RXOVERR) != RESET)         //接收fifo上溢错误
                {
                         SDIO_ClearFlag(SDIO_FLAG_RXOVERR);                //清错误标志
                        return SD_RX_OVERRUN;                 
                }else if(SDIO_GetFlagStatus(SDIO_FLAG_STBITERR) != RESET)         //接收起始位错误
                {
                         SDIO_ClearFlag(SDIO_FLAG_STBITERR);//清错误标志
                        return SD_START_BIT_ERR;                 
                }  
        SDIO_ClearFlag(SDIO_STATIC_FLAGS);//清除所有标记
        //把数据顺序按8位为单位倒过来.           
        *(pscr+1)=((tempscr[0]&SD_0TO7BITS)<<24)|((tempscr[0]&SD_8TO15BITS)<<8)|((tempscr[0]&SD_16TO23BITS)>>8)|((tempscr[0]&SD_24TO31BITS)>>24);
        *(pscr)=((tempscr[1]&SD_0TO7BITS)<<24)|((tempscr[1]&SD_8TO15BITS)<<8)|((tempscr[1]&SD_16TO23BITS)>>8)|((tempscr[1]&SD_24TO31BITS)>>24);
         return errorstatus;
}

友情提示: 此问题已得到解决,问题已经关闭,关闭后问题禁止继续编辑,回答。
该问题目前已经被作者或者管理员关闭, 无法添加新回复
12条回答
ZDawn
1楼-- · 2019-07-21 08:24
正点原子 发表于 2018-1-10 00:24
你是用的我们开发板么?直接下载我们例程,不要做任何修改,测试看看?试试寄存器版本?

我在板子上电3秒之后再去初始化SD卡结果就可以了,可能是我这张SD卡需要一定的电压稳定时间,或者是板子电路问题
正点原子
2楼-- · 2019-07-21 12:56
你是用的我们开发板么?直接下载我们例程,不要做任何修改,测试看看?试试寄存器版本?
ZDawn
3楼-- · 2019-07-21 15:45
板子我试过硬件复位的方法,也不能使程序正常。一定要用KEIL 在线调试的那个复位按钮来复位程序才正常。。。
ZDawn
4楼-- · 2019-07-21 16:56
我在没有插SD卡的情况下用这条语“while(SD_Init());”去运行,程序不停地执行初始化,中途再插入SD卡的话也会卡死在上面程序的死循环中。
ZDawn
5楼-- · 2019-07-21 17:38
 精彩回答 2  元偷偷看……
fEndman
6楼-- · 2019-07-21 21:49
和你的情况完全一样,我也是16gb卡,卡在FindSCR那个循环。。。我现在又发现如果断开调试让程序单独运行是可以成功初始化的,请问楼主这个问题解决没有?

一周热门 更多>