当前位置:首页 > 单片机 > 单片机
[导读] 一、uboot中SD卡的初始化二、读取扇区三、 写变量到SD卡中四、一、uboot中SD卡的初始化1.1 硬件连线MMC0_CDN-->GPG6 -->SD卡检测引脚MMC0_WPN-->GPGL13 -->SD卡写保护引脚MMC0_DATA[0-3]--> GPG[2-5] -

一、uboot中SD卡的初始化
二、读取扇区
三、 写变量到SD卡中
四、

一、uboot中SD卡的初始化
1.1 硬件连线

  1. MMC0_CDN-->GPG6 -->SD卡检测引脚

  2. MMC0_WPN-->GPGL13 -->SD卡写保护引脚

  3. MMC0_DATA[0-3]--> GPG[2-5] -->

  4. MMC0_CMD -->GPG1 -->SD命令线

  5. MMC0_CLK -->GPG0 -->SDIO/SD卡时钟线

1.2 uboot中初始化过程
uboot lib_arm/board.c中

  1. void start_armboot (void)

  2. {

  3. #ifdefined(CONFIG_BOOT_MOVINAND)

  4. puts("SD/MMC: ");

  5. if((0x24564236==magic[0])&&(0x20764316==magic[1])){

  6. printf("Boot up for burningn");

  7. }else{

  8. movi_set_capacity();

  9. movi_set_ofs(MOVI_TOTAL_BLKCNT);

  10. movi_init();

  11. }

  12. #endif

  13. }

a. 设置为sector mode

  1. movi_set_capacity();

  2. {

  3. if(MOVI_HIGH_CAPACITY&0x1)//insector mode,this value will beset

  4. movi_hc=1;//全局变量movi_hc=1

  5. }

b.以扇区为单位,设置BL1,ENV,BL2,kernel,rootfs的位置

  1. movi_set_ofs(MOVI_TOTAL_BLKCNT);

  2. {

  3. if(ofsinfo.last!=last){

  4. ofsinfo.last=last-(eFUSE_SIZE/MOVI_BLKSIZE);

  5. ofsinfo.bl1=ofsinfo.last-MOVI_BL1_BLKCNT;

  6. ofsinfo.env=ofsinfo.bl1-MOVI_ENV_BLKCNT;

  7. ofsinfo.bl2=ofsinfo.bl1-(MOVI_BL2_BLKCNT+MOVI_ENV_BLKCNT);

  8. ofsinfo.kernel=ofsinfo.bl2-MOVI_ZIMAGE_BLKCNT;

  9. ofsinfo.rootfs=ofsinfo.kernel-MOVI_ROOTFS_BLKCNT;

  10. changed=1;

  11. }

  12. }

c. SD控制器初始化

  1. movi_init();

  2. {

  3. hsmmc_set_gpio();

  4. {

  5. reg=readl(GPGCON)&0xf0000000;

  6. writel(reg|0x02222222,GPGCON);//设置 GPG为MMC模式

  7. reg=readl(GPGPUD)&0xfffff000;

  8. writel(reg,GPGPUD);//上拉电阻disable

  9. }

  10. hsmmc_reset();

  11. {

  12. s3c_hsmmc_writeb(0x3,HM_SWRST); //data_line && cmd_line reset

  13. }

  14. if(hsmmc_init()){

  15. printf("nCard Initialization failed.n");

  16. return-1;

  17. }

  18. return 1;

  19. }

cpu/s3c64xx/hs_mmc.c

  1. inthsmmc_init(void)

  2. {

  3. hsmmc_clock_onoff(0);//关闭SDCLK

  4. {

  5. a.CLKCON0[2]:0//SD Clock diaable

  6. }


  7. a.SCLK_GATE-->0x7E00_F038[27]:

  8. SCLK_MMC0_48[27]Gating special clockforMMC0(0:mask,1:pass)


  9. b.

  10. set_clock(SD_EPLL,0x80);

  11. {

  12. //CONTROL2_0 0x7C200080 R/W Control register 2(Channel 0)

  13. //[5:4]:Base Clock SourceSelect-->EPLL

  14. //[14]:1 Feedback Clock EnableforRx Data/Command Clock

  15. //[30]:1 Command Conflict Mask Enable

  16. //[31]:1 Write Status Clear Async Mode Enable

  17. s3c_hsmmc_writel(0xC0004100|(clksrc<<4),HM_CONTROL2);


  18. s3c_hsmmc_writel(0x00008080,HM_CONTROL3);

  19. s3c_hsmmc_writel(0x3<<16,HM_CONTROL4);

  20. //下3条:SDCLK FrequencySelect

  21. //div=80h base clock divided by 256

  22. //

  23. //最后的for The SD Host Driver shall waittosetSD Clock Enableuntilthis bitissetto1.

  24. s3c_hsmmc_writew(s3c_hsmmc_readw(HM_CLKCON)&~(0xff<<8),HM_CLKCON);

  25. s3c_hsmmc_writew(((div<<8)|0x1),HM_CLKCON);

  26. for(i=0;i<0x10000;i++);//wait stable


  27. hsmmc_clock_onoff(1);

  28. }

  29. c.s3c_hsmmc_writeb(0xe,HM_TIMEOUTCON);

  30. TIME_OUT_CNT=1110b TMCLK x 2的27

  31. }

为了区别SD卡是2.0还是1.0,或是MMC卡,这里根据协议向上兼容的原理,首先发送只有SD2.0才有的命令CMD8,如果CMD8返回无错误,则初步判断为2.0卡,进一步发送命令循环发送CMD55+ACMD41,直到返回0x00,确定SD2.0卡初始化成功,进入Ready状态,再发送CMD58命令来判断是HCSD还是SCSD,到此SD2.0卡初始化成功。如果CMD8返回错误则进一步判断为1.0卡还是MMC卡,循环发送CMD55+ACMD41,返回无错误,则为SD1.0卡,到此SD1.0卡初始成功,如果在一定的循环次数下,返回为错误,则进一步发送CMD1进行初始化,如果返回无错误,则确定为MMC卡,如果在一定的次数下,返回为错误,则不能识别该卡,初始结束。


二、读取扇区
SDMK6410> fatload mmc 0:1 50008000 zImage
以读取MBR为例
do_fat_fsload
--> fat_register_device(dev_desc,part);
--> dev_desc->block_read(dev_desc->dev, 0, 1, (ulong *)buffer);
1. 初始化时
hsmmc_init() //初始化mmc_dev的函数指针
{
mmc_dev.if_type = IF_TYPE_MMC;
mmc_dev.part_type = PART_TYPE_DOS;
mmc_dev.dev = 0;
mmc_dev.blksz = MMC_BLOCK_SIZE;
mmc_dev.block_read = mmc_bread;
}
2. 正式进入读取部分
cpu/s3c64xx/hs_mmc.c L1337
/*
dev_num: 第几个mmc设备
blknr: 开始的扇区
blkcnt: 要读取的扇区数
dst: 数据保存到的dst_buffer
*/
static ulong mmc_bread (int dev_num, ulong blknr, ulong blkcnt, ulong* dst)
{
if (dst >= 0xc0000000)
dst = virt_to_phys(dst); //取物理地址
if (blkcnt != 0) //连dev_num都不要了,看来只能支持1个SD设备
movi_read((uint) dst, (uint) blknr, (uint) blkcnt);
return blkcnt;
}

void movi_read (uint addr, uint start_blk, uint blknum)
{
}

三、写入扇区
SMDK6410> saveenv


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

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 隧道灯 驱动电源
关闭