当前位置:首页 > 单片机 > 单片机
[导读]/*---------------------------------------------------------------文件名称:NBOOT.c说 明:进行Nand Bootloader操作---------------------------------------------------------------*/#include "S3C244x.h"#inc

/*
---------------------------------------------------------------
文件名称:NBOOT.c
说 明:进行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字节
#define SECTOR_PER_BLOCK 256 //每块含有多少页

/******************************************************
*文件名称:ReadNandImageToRam
*输 入:unEntry 进入点
*输 出:WINCE镜像跳转地址
0为无效
*功能说明:从NAND FLASH加载WINCE镜像到SDRAM
*******************************************************/
UINT32 ReadNandImageToRam(UINT32 unEntry)
{
UINT32 unSectorsNeeded;
UINT32 unSector,unLength;
UINT32 unRam,i;


PTOC pTOC=(PTOC)0x33F00000;//若用TOC类型定义,不需要自行赋给地址
//若用PTOC类型定义,编译器给予pTOC变量地址是未知的
//因此要强制赋给其地址
NandReadSector(TOC_BLOCK*SECTOR_PER_BLOCK,(UINT8 *)pTOC);//读取第512扇区内容,并赋给TOC

if(!TOCIsValid(pTOC))
{
DEBUGMSG("rnTOC is not validrn");
return 0;
}
DEBUGMSG("rnTOC is validrn");
//参数dwEntry 定义入口方式,是直接进入WinCE内核,还是启动Eboot
if(!(pTOC->id[unEntry].dwImageType &IMAGE_TYPE_RAMIMAGE))
{
return 0;
}

DEBUGMSG("rnImage is RamImagern");

unSectorsNeeded= pTOC->id[unEntry].dwTtlSectors; //镜像占用的扇区总数
unRam=VIRTUAL_TO_PHYSICAL(pTOC->id[unEntry].dwLoadAddress);//虚拟地址转变为物理地址

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;//不进行下面的操作
}

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) ://获取加载镜像完成后的跳转地址
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内存空间的注解
在WINCE的EBOOT中
ARGS 80028000 00000800 RESERVED
自0x80028000处,大小为0x00000800的空间被命名为ARGS,这个空间类型为
RESERVED,用于保存Bootloader和操作系统之间的共享数据

总之在0x30020800地址处必须要对其进行初始化为0,否则WINCE不能够正常激活
以下选自2440的BSP上的代码。

在BSP中,boot loader和OAL之间有一些数据可以共享
这段数据被存放在(BSP_ARGS *) IMAGE_SHARE_ARGS_UA_START这块内存里。
#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指向的是虚拟内存中的Uncached段,而IMAGE_SHARE_ARGS_CA_START指向虚拟内存中的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对应的物理地址的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);//关于pBSPArgs的地址与长度需要根据BSP包进行确定的
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;
}

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

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