当前位置:首页 > 单片机 > 单片机
[导读]问题描述:    TQ2440的官方裸跑程序中,对SD卡先进行读操作,然后再写,发现不能程序卡死。倘若对SD卡先写后读,程序可以正常运行,奇哉怪哉?写数据的关键代码-->while(i SDCard_BlockSize) + 1) &0x0fff; r

问题描述:  

  TQ2440的官方裸跑程序中,对SD卡先进行读操作,然后再写,发现不能程序卡死。倘若对SD卡先写后读,程序可以正常运行,奇哉怪哉?

写数据的关键代码-->

while(i


调试与问题分析:

  调试的时候发现,当不能在写的时候,FIFO available detectfor Tx (TFDET)为0,也即是说是fifo满了。此时,程序循环了16次(i=0x10)。循环一次写入4个字节,16次刚好是fifo的最大容量64字节。这证明了,写入fifo中的数据,本应该发送给SD卡,腾空fifo以供用户继续写,却被搁置在fifo中,有进无出。就像下水道中转站被堵,上游的污水就不能继续排放一行的道理。

先写后读是可以正常工作的,我打印了执行写函数之后部分寄存器的值,如左图所示。可以发现写后的寄存器rSDIDCNT、rSDIDSTA都恢复到了初始值。右图是执行读函数之后寄存器的值,可以发现执行读函数之后,rSDIDCNT、rSDIDSTA都没有回到初始值,都仍然停留在读函数执行的状态中。也就是说,读函数没有执行彻底,SDMMC模块没有进入到空闲状态。在没有准备好的情况下,继续进行写操作,是不可能成功的。

修复

  修复的方法主要是无论读操作,还是写操作,都确认SDIO总线空闲时,然后再才退出当前的函数。这样可以保证在随后的操作中,SDMMC模块处于准备好的状态,而非遗留状态。

读函数

/**********************************************************************************

功 能:该函数用于从SD卡中读出指定块起始地址的单个数据块

参 数:

U32 Addr 被读块的起始地址

U8* RxBuffer 用于接收读出数据的缓冲区

返回值:

0 读块操作不成功

1 读块操作成功

举 例:

在主调函数中定义一个数组作为接收缓冲区,如U8 Rx_buffer[BlockSize];

然后开始调用Read_One_Block(addr,Rx_buffer);

**********************************************************************************/

U8 Read_One_Block(U32 Addr,U8 * RxBuffer)

{

U16 i=0;

U32 status=0;

U16 BlockSize; //定义块大小

U16 BlockNumber;

BlockSize=1 << SDCard_BlockSize; //以byte为单位

BlockNumber = ((Addr >> SDCard_BlockSize) + 1) &0x0fff;

rSDIDTIMER=0x7fffff; // Set timeout count

rSDIBSIZE=0x200; // 512byte(128word)

rSDIFSTA=rSDIFSTA"(1<<16); // FIFO reset

rSDIDCON = (BlockNumber<<0)|(2<<12)|(1<<14)|(1<<16)|(1<<17)|(1<<19)|(0<<22);

while(CMD17(Addr)!=1) //发送读单个块指令

{

#ifdef __SD_MMC_DEBUG__

Uart_Printf("Send read addr failed!n");

#endif

}

/* 开始接收数据到缓冲区 */

while(i

{

status = rSDIDSTA;

if(status&0x60) //检查是否超时和CRC校验是否出错

{

rSDIDSTA=(0x3<<0x5); //清除超时标志和CRC错误标志

#ifdef __SD_MMC_DEBUG__

Uart_Printf("there is wrong when reading: %s.n",status&0x20 ? "time out" :"CRC error");

#endif

return 0;

}

status=rSDIFSTA;

if((status&0x1000)==0x1000) //如果接收FIFO中有数据

{

*RxBuffer=rSDIDAT;

RxBuffer++;

i++;

}

status = rSDIDSTA;

Delay(2); //延时2ms

rSDIDCON=rSDIDCON&~(7<<12); //结束SDMMC模块的接收

rSDIDSTA = status; //清状态标志位


/* 确认SD卡进入了空闲状态--SDIO总线空闲 */

rSDIDCON=(0<<18)|(1<<17)|(1<<16)|(1<<14)|(1<<12)|(BlockNumber<<0);

rSDIDTIMER=0x7fffff;

status = rSDIDSTA;

while( !( ((status&0x08)==0x08) | ((status&0x20)==0x20)| ((status&0x800)==0x800) )){

status=rSDIDSTA;

}

if( (status&0x20) == 0x20 ){

rSDIDSTA = status;

return 0;

}

rSDIDSTA = status;

return 1;

}


写函数

/**********************************************************************************

功 能:该函数用于向SD卡的一个数据块写入数据

参 数:

U32 Addr 被写块的起始地址

U8* TxBuffer 用于发送数据的缓冲区

返回值:

0 数据写入操作失败

1 数据写入操作成功

举 例:

在主调函数中定义一个数组作为发送缓冲区,如U8 Tx_buffer[BlockSize];

然后开始调用Write_One_Block(addr,Tx_buffer);

**********************************************************************************/

U8 Write_One_Block(U32 Addr,const U8 * TxBuffer)

{

U16 i = 0;

U32 status = 0;

U16 BlockSize; //定义块大小

U16 BlockNumber;

BlockSize = 1<< SDCard_BlockSize; //以byte为单位

BlockNumber = ((Addr >> SDCard_BlockSize) +1) &0x0fff;

rSDIDTIMER=0x7fffff; // Set timeout count

rSDIBSIZE=0x200; // 512 byte(128 word)

rSDIFSTA = rSDIFSTA|(1<<16); // FIFO reset

rSDIDCON = BlockNumber|(3<<12)|(1<<14)|(1<<16)|(1<<17)|(1<<20)|(0<<22);


while(CMD24(Addr)!=1) //发送写单块操作指令

{

#ifdef __SD_MMC_DEBUG__

Uart_Printf("Send write addr failed!n");

#endif

}

/* 开始传递数据到缓冲区 */

while(i < BlockSize)

{

status=rSDIFSTA;

if((status&0x2000)==0x2000) //如果发送FIFO可用,即FIFO未满

{

rSDIDAT = *TxBuffer;

TxBuffer++;

i++;

}

}

status = rSDIDSTA;

Delay(5);

rSDIDCON=rSDIDCON&~(7<<12); //结束SDMMC模块的发送

rSDIDSTA = status;

/* 确认SD卡进入了空闲状态--SDIO总线空闲 */

rSDIDCON=(0<<18)|(1<<17)|(1<<16)|(1<<14)|(1<<12)|(BlockNumber<<0);

rSDIDTIMER=0x7fffff; // Set timeout count

status = rSDIDSTA;

while( !( ((status&0x08)==0x08) | ((status&0x20)==0x20)| ((status&0x800)==0x800) )){

status=rSDIDSTA;

}

if( (status&0x20) == 0x20 ){

rSDIDSTA = status;

return 0;

}

rSDIDSTA = status;

return 1;

}


测试效果

  以下操作都得到成功验证:

单块读

单块写

多块读(调用单块读函数实现)

多块写(调用单块写函数实现)

先读后写

先写后读

 移植fatfs文件系统测试:

  大多数操作没有故障出现,偶尔也会出现写函数卡死的情况

仍然存在

本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: 驱动电源

在工业自动化蓬勃发展的当下,工业电机作为核心动力设备,其驱动电源的性能直接关系到整个系统的稳定性和可靠性。其中,反电动势抑制与过流保护是驱动电源设计中至关重要的两个环节,集成化方案的设计成为提升电机驱动性能的关键。

关键字: 工业电机 驱动电源

LED 驱动电源作为 LED 照明系统的 “心脏”,其稳定性直接决定了整个照明设备的使用寿命。然而,在实际应用中,LED 驱动电源易损坏的问题却十分常见,不仅增加了维护成本,还影响了用户体验。要解决这一问题,需从设计、生...

关键字: 驱动电源 照明系统 散热

根据LED驱动电源的公式,电感内电流波动大小和电感值成反比,输出纹波和输出电容值成反比。所以加大电感值和输出电容值可以减小纹波。

关键字: LED 设计 驱动电源

电动汽车(EV)作为新能源汽车的重要代表,正逐渐成为全球汽车产业的重要发展方向。电动汽车的核心技术之一是电机驱动控制系统,而绝缘栅双极型晶体管(IGBT)作为电机驱动系统中的关键元件,其性能直接影响到电动汽车的动力性能和...

关键字: 电动汽车 新能源 驱动电源

在现代城市建设中,街道及停车场照明作为基础设施的重要组成部分,其质量和效率直接关系到城市的公共安全、居民生活质量和能源利用效率。随着科技的进步,高亮度白光发光二极管(LED)因其独特的优势逐渐取代传统光源,成为大功率区域...

关键字: 发光二极管 驱动电源 LED

LED通用照明设计工程师会遇到许多挑战,如功率密度、功率因数校正(PFC)、空间受限和可靠性等。

关键字: LED 驱动电源 功率因数校正

在LED照明技术日益普及的今天,LED驱动电源的电磁干扰(EMI)问题成为了一个不可忽视的挑战。电磁干扰不仅会影响LED灯具的正常工作,还可能对周围电子设备造成不利影响,甚至引发系统故障。因此,采取有效的硬件措施来解决L...

关键字: LED照明技术 电磁干扰 驱动电源

开关电源具有效率高的特性,而且开关电源的变压器体积比串联稳压型电源的要小得多,电源电路比较整洁,整机重量也有所下降,所以,现在的LED驱动电源

关键字: LED 驱动电源 开关电源

LED驱动电源是把电源供应转换为特定的电压电流以驱动LED发光的电压转换器,通常情况下:LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: LED 隧道灯 驱动电源
关闭