S3C6410使用---7uboot中SD初始化及讀寫分析
一、uboot中SD卡的初始化
二、讀取扇區(qū)
三、 寫變量到SD卡中
四、
一、uboot中SD卡的初始化
1.1 硬件連線
MMC0_CDN-->GPG6 -->SD卡檢測引腳
MMC0_WPN-->GPGL13 -->SD卡寫保護(hù)引腳
MMC0_DATA[0-3]--> GPG[2-5] -->
MMC0_CMD -->GPG1 -->SD命令線
MMC0_CLK -->GPG0 -->SDIO/SD卡時鐘線
1.2 uboot中初始化過程
uboot lib_arm/board.c中
void start_armboot (void)
{
#ifdefined(CONFIG_BOOT_MOVINAND)
puts("SD/MMC: ");
if((0x24564236==magic[0])&&(0x20764316==magic[1])){
printf("Boot up for burningn");
}else{
movi_set_capacity();
movi_set_ofs(MOVI_TOTAL_BLKCNT);
movi_init();
}
#endif
}
a. 設(shè)置為sector mode
movi_set_capacity();
{
if(MOVI_HIGH_CAPACITY&0x1)//insector mode,this value will beset
movi_hc=1;//全局變量movi_hc=1
}
b.以扇區(qū)為單位,設(shè)置BL1,ENV,BL2,kernel,rootfs的位置
movi_set_ofs(MOVI_TOTAL_BLKCNT);
{
if(ofsinfo.last!=last){
ofsinfo.last=last-(eFUSE_SIZE/MOVI_BLKSIZE);
ofsinfo.bl1=ofsinfo.last-MOVI_BL1_BLKCNT;
ofsinfo.env=ofsinfo.bl1-MOVI_ENV_BLKCNT;
ofsinfo.bl2=ofsinfo.bl1-(MOVI_BL2_BLKCNT+MOVI_ENV_BLKCNT);
ofsinfo.kernel=ofsinfo.bl2-MOVI_ZIMAGE_BLKCNT;
ofsinfo.rootfs=ofsinfo.kernel-MOVI_ROOTFS_BLKCNT;
changed=1;
}
}
c. SD控制器初始化
movi_init();
{
hsmmc_set_gpio();
{
reg=readl(GPGCON)&0xf0000000;
writel(reg|0x02222222,GPGCON);//設(shè)置 GPG為MMC模式
reg=readl(GPGPUD)&0xfffff000;
writel(reg,GPGPUD);//上拉電阻disable
}
hsmmc_reset();
{
s3c_hsmmc_writeb(0x3,HM_SWRST); //data_line && cmd_line reset
}
if(hsmmc_init()){
printf("nCard Initialization failed.n");
return-1;
}
return 1;
}
cpu/s3c64xx/hs_mmc.c
inthsmmc_init(void)
{
hsmmc_clock_onoff(0);//關(guān)閉SDCLK
{
a.CLKCON0[2]:0//SD Clock diaable
}
a.SCLK_GATE-->0x7E00_F038[27]:
SCLK_MMC0_48[27]Gating special clockforMMC0(0:mask,1:pass)
b.
set_clock(SD_EPLL,0x80);
{
//CONTROL2_0 0x7C200080 R/W Control register 2(Channel 0)
//[5:4]:Base Clock SourceSelect-->EPLL
//[14]:1 Feedback Clock EnableforRx Data/Command Clock
//[30]:1 Command Conflict Mask Enable
//[31]:1 Write Status Clear Async Mode Enable
s3c_hsmmc_writel(0xC0004100|(clksrc<<4),HM_CONTROL2);
s3c_hsmmc_writel(0x00008080,HM_CONTROL3);
s3c_hsmmc_writel(0x3<<16,HM_CONTROL4);
//下3條:SDCLK FrequencySelect
//div=80h base clock divided by 256
//
//最后的for The SD Host Driver shall waittosetSD Clock Enableuntilthis bitissetto1.
s3c_hsmmc_writew(s3c_hsmmc_readw(HM_CLKCON)&~(0xff<<8),HM_CLKCON);
s3c_hsmmc_writew(((div<<8)|0x1),HM_CLKCON);
for(i=0;i<0x10000;i++);//wait stable
hsmmc_clock_onoff(1);
}
c.s3c_hsmmc_writeb(0xe,HM_TIMEOUTCON);
TIME_OUT_CNT=1110b TMCLK x 2的27
}
為了區(qū)別SD卡是2.0還是1.0,或是MMC卡,這里根據(jù)協(xié)議向上兼容的原理,首先發(fā)送只有SD2.0才有的命令CMD8,如果CMD8返回?zé)o錯誤,則初步判斷為2.0卡,進(jìn)一步發(fā)送命令循環(huán)發(fā)送CMD55+ACMD41,直到返回0x00,確定SD2.0卡初始化成功,進(jìn)入Ready狀態(tài),再發(fā)送CMD58命令來判斷是HCSD還是SCSD,到此SD2.0卡初始化成功。如果CMD8返回錯誤則進(jìn)一步判斷為1.0卡還是MMC卡,循環(huán)發(fā)送CMD55+ACMD41,返回?zé)o錯誤,則為SD1.0卡,到此SD1.0卡初始成功,如果在一定的循環(huán)次數(shù)下,返回為錯誤,則進(jìn)一步發(fā)送CMD1進(jìn)行初始化,如果返回?zé)o錯誤,則確定為MMC卡,如果在一定的次數(shù)下,返回為錯誤,則不能識別該卡,初始結(jié)束。
二、讀取扇區(qū)
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的函數(shù)指針
{
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. 正式進(jìn)入讀取部分
cpu/s3c64xx/hs_mmc.c L1337
/*
dev_num: 第幾個mmc設(shè)備
blknr: 開始的扇區(qū)
blkcnt: 要讀取的扇區(qū)數(shù)
dst: 數(shù)據(jù)保存到的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設(shè)備
movi_read((uint) dst, (uint) blknr, (uint) blkcnt);
return blkcnt;
}
void movi_read (uint addr, uint start_blk, uint blknum)
{
}
三、寫入扇區(qū)
SMDK6410> saveenv