记录下 分散加载LPC1788

 这个链接是我在官网上找到的关于分散加载文件的资料。讲的比较详细了。这里通过一个例子记录下我学习的过程,通过分散加载文件把代码从flash里拷贝到ram里运行, 基于LPC1788。

    先贴下我的sct文件:

LR_IROM1 0x00000000 0x00002000  
{ 
	ER_IROM1 0x00000000 0x00020000  
	{
		*.o (RESET, +First)
		*(InRoot$$Sections)
		startup_lpc177x_8x.o (+RO)
		system_LPC177x_8x.o (+RO)
	}
	
	RW_IRAM1 0x20000000 0x00004000  
	{
		.ANY (+RW +ZI)
	}
}

LR_IROM2 0x00002000	0x0007E000
{
	VECTOR 0x10000000 EMPTY 0xE4
	{
	}
	
	ER_IRAM1 +0
	{
		.ANY (+RO)
	}
}


这里有两个加载域(load region)LR_IROM1和LR_IROM2,LR_IROM1是初始化程序,拷贝代码等,从ROM的地址0开始,LR_ROM2是应用程序,从ROM的0x2000开始。+RO表示只读,代码或者只读数据,一般用来表示代码,+RW表示可读可写的数据,+ZI表示初始化为0的数据。大括号里面的为运行域(execution region),一个加载域可以包含几个运行域,LR_ROM2里面有两个运行域,VECTOR和ER_IRAM1,我用VECTOR来表示中断向量区域,ER_IRAM1来表示应用程序区,+0表示紧接着VECTOR排放,EMPTY表示空的,这里空出0xE4的大小,用来放中断向量,.ANY表示除了上面用到的代码之外的代码,官网上有专门解释.ANY的一节。

    下面用一张图来表示这个程序的加载域和执行域:

 

其实加载域的empty这块区域是不用空出来的,主要是运行域要空出来,用来拷贝中断向量,看个人喜好了,我觉得空出来方便引用这块区域的执行域地址。

    这样框架就比较清楚了,拷贝的程序清单如下:

extern unsigned char Image$$VECTOR$$Base;
extern unsigned char Image$$VECTOR$$Length;

extern unsigned char Load$$ER_IRAM1$$Base;
extern unsigned char Image$$ER_IRAM1$$Base;
extern unsigned char Image$$ER_IRAM1$$Length;

void CopyCode2Ram ()
{
    unsigned char *pSrc, *pDes;
    unsigned int count;
    
    SCB->VTOR = 0x10000000;
    
    pSrc = 0;
    pDes = (unsigned char*)&Image$$VECTOR$$Base;
    count = 0xE4;
    
    while (count--)
    {
        *pDes++ = *pSrc++;
    }
    
    
    count = (unsigned int)&Image$$ER_IRAM1$$Length;
    pDes = (unsigned char*)&Image$$ER_IRAM1$$Base;
    pSrc = (unsigned char*)(&Load$$ER_IRAM1$$Base + 0xE4);
    
    while (count--)
    {
        *pDes++ = *pSrc++;
    }
}


其中拷贝中断向量的时候要指定中断向量的偏移地址。Load

ERIRAM1ERIRAM1

Base表示执行域ER_IRAM1的加载地址;Image

ERIRAM1ERIRAM1

Base表示执行域ER_IRAM1的执行地址;Image

ERIRAM1ERIRAM1

Length表示执行域ER_IRAM1的实际长度,VECTOR区域因为是EMPTY,所以实际长度是0,而中断向量的长度是固定的,所以程序里就写了个常数。

猜你喜欢

转载自blog.csdn.net/huan447882949/article/details/81081966
今日推荐