/*
---------------------------------------------------------------
文件名稱:NBOOT.c
說 明:進(jìn)行Nand Bootloader操作
---------------------------------------------------------------
*/
#include "S3C244x.h"
#include "UART.h"
#include "Global.h"
#include "TOC.h"
#include "Nand.h"
#include "MMU.h"
#define TOC_BLOCK (2) //TOC占用2塊
#define SECTOR_SIZE 512 //頁大小:512字節(jié)
#define SECTOR_PER_BLOCK 256 //每塊含有多少頁
/******************************************************
*文件名稱:ReadNandImageToRam
*輸 入:unEntry 進(jìn)入點(diǎn)
*輸 出:WINCE鏡像跳轉(zhuǎn)地址
0為無效
*功能說明:從NAND FLASH加載WINCE鏡像到SDRAM
*******************************************************/
UINT32 ReadNandImageToRam(UINT32 unEntry)
{
UINT32 unSectorsNeeded;
UINT32 unSector,unLength;
UINT32 unRam,i;
PTOC pTOC=(PTOC)0x33F00000;//若用TOC類型定義,不需要自行賦給地址
//若用PTOC類型定義,編譯器給予pTOC變量地址是未知的
//因此要強(qiáng)制賦給其地址
NandReadSector(TOC_BLOCK*SECTOR_PER_BLOCK,(UINT8 *)pTOC);//讀取第512扇區(qū)內(nèi)容,并賦給TOC
if(!TOCIsValid(pTOC))
{
DEBUGMSG("rnTOC is not validrn");
return 0;
}
DEBUGMSG("rnTOC is validrn");
//參數(shù)dwEntry 定義入口方式,是直接進(jìn)入WinCE內(nèi)核,還是啟動Eboot
if(!(pTOC->id[unEntry].dwImageType &IMAGE_TYPE_RAMIMAGE))
{
return 0;
}
DEBUGMSG("rnImage is RamImagern");
unSectorsNeeded= pTOC->id[unEntry].dwTtlSectors; //鏡像占用的扇區(qū)總數(shù)
unRam=VIRTUAL_TO_PHYSICAL(pTOC->id[unEntry].dwLoadAddress);//虛擬地址轉(zhuǎn)變?yōu)槲锢淼刂?br/>
DEBUGMSGEx("rnSectors Total=",pTOC->id[unEntry].dwTtlSectors ,10);
DEBUGMSGEx("rnVirtual Load Address =",pTOC->id[unEntry].dwLoadAddress,16);
DEBUGMSGEx("rnPhysical Load Address =",unRam,16);
i=0;
while(unSectorsNeeded && i
unSector = pTOC->id[unEntry].sgList[i].dwSector;
unLength = pTOC->id[unEntry].sgList[i].dwLength;
while(unLength)
{
if(unSector % SECTOR_PER_BLOCK == 0)//256頁檢查一次壞塊
{
if(!NandCheckBadBlock(unSector/SECTOR_PER_BLOCK))
{
unSector+=SECTOR_PER_BLOCK;//存在壞塊
continue;//不進(jìn)行下面的操作
}
DEBUGMSG(".");
}
if(!NandReadSector(unSector,(UINT8 *)unRam))
{
DEBUGMSG("rnCan't Load WINCE Imagern");
while(1);
}
unSector++;
unLength--;
unRam+=SECTOR_SIZE;
}
unSectorsNeeded-=pTOC->id[unEntry].sgList[i].dwLength;
i++;
}
return (pTOC->id[unEntry].dwJumpAddress?VIRTUAL_TO_PHYSICAL(pTOC->id[unEntry].dwJumpAddress) ://獲取加載鏡像完成后的跳轉(zhuǎn)地址
VIRTUAL_TO_PHYSICAL(pTOC->id[unEntry].dwLoadAddress));
}
void PORTInit(void)
{
rGPACON = 0x7fffff;
rGPDCON = 0xaaaaaaaa;
rGPDUP = 0xffff; // The pull up function is disabled GPD[15:0]
rGPECON = 0xaaaaaaaa;
rGPEUP = 0xffff;
rGPFUP = 0xff;
rGPGCON = 0xFD000000;
rGPGUP = 0xffff;
rGPHCON = 0x02faaa;
rGPHUP = 0x7ff;
rEXTINT0 = 0x22222222; // EINT[7:0]
rEXTINT1 = 0x22222222; // EINT[15:8]
rEXTINT2 = 0x22222222; // EINT[23:16]
}
#define BSP_ARGS_ADDR 0x30020800
#define BSP_ARGS_SIZE 0x800
/*
ARGS內(nèi)存空間的注解
在WINCE的EBOOT中
ARGS 80028000 00000800 RESERVED
自0x80028000處,大小為0x00000800的空間被命名為ARGS,這個空間類型為
RESERVED,用于保存Bootloader和操作系統(tǒng)之間的共享數(shù)據(jù)
總之在0x30020800地址處必須要對其進(jìn)行初始化為0,否則WINCE不能夠正常激活
以下選自2440的BSP上的代碼。
在BSP中,boot loader和OAL之間有一些數(shù)據(jù)可以共享
這段數(shù)據(jù)被存放在(BSP_ARGS *) IMAGE_SHARE_ARGS_UA_START這塊內(nèi)存里。
#define IMAGE_SHARE_ARGS_UA_START 0xA0020000
#define IMAGE_SHARE_ARGS_CA_START 0x80020800
#define IMAGE_SHARE_ARGS_SIZE 0x00000800
可以看出來,IMAGE_SHARE_ARGS_UA_START指向的是虛擬內(nèi)存中的Uncached段,而IMAGE_SHARE_ARGS_CA_START指向虛擬內(nèi)存中的Cached段。
我的g_oalAddressTable中這樣定義:
DCD 0x80000000, 0x30000000, 64 ; 32 MB DRAM BANK 6
因此,它們分別是指向物理地址0x30020000和0x30020800。
然后看config.bib中,可以看到有這段定義:
; Common RAM areas
AUD_DMA 80002000 00000800 RESERVED
SDIO_DMA 80010000 00010000 RESERVED
ARGS 80020800 00000800 RESERVED
DBGSER_DMA 80022000 00002000 RESERVED
SER_DMA 80024000 00002000 RESERVED
IR_DMA 80026000 00002000 RESERVED
SLEEP 80028000 00002000 RESERVED
EDBG 80030000 00020000 RESERVED
DISPLAY 80100000 00100000 RESERVED
其中ARGS對應(yīng)的物理地址的0x30020800是為IMAGE_SHARE_ARGS_CA_START保留的吧。
*/
int Main(void)
{
void (*RunWINCE)(void)=0;
MMUEnableICache();
MMUEnableDCache();
PORTInit(); //IO口一定要初始化,否則串口不能正常工作
UARTInit(S3C2440PCLK,115200);
NandInit();
DEBUGMSGEx("rnrnPhysical BSPArgs Address = ",BSP_ARGS_ADDR,16);
DEBUGMSGEx("rnBSPArgs Size = ",BSP_ARGS_SIZE,16);
DEBUGMSG("rnClearing BSPArgs Memoryrn");
BufClr((char*)BSP_ARGS_ADDR, BSP_ARGS_SIZE);//關(guān)于pBSPArgs的地址與長度需要根據(jù)BSP包進(jìn)行確定的
DEBUGMSG("rnLoading WINCErn");
RunWINCE=(void(*)(void))ReadNandImageToRam(1);
DEBUGMSGEx("rnPhysical Jump Address =",(UINT32)RunWINCE,16);
DEBUGMSGEx("rn Virtual Jump Address =",PHYSICAL_TO_VIRTUAL((UINT32)RunWINCE),16);
if(RunWINCE)
{
DEBUGMSG("rnLoad WINCE Successfullyrn");
RunWINCE();
}
while(1);
return 0;
}