搜索
bottom↓
回复: 22

【分享】IAR编译环境之:ILINK链接器的.ICF文件编写——转帖

[复制链接]

出0入0汤圆

发表于 2014-9-3 13:25:25 | 显示全部楼层 |阅读模式
1.首先说说什么是icf文件(即ILINK链接器的配置文件)的作用,其实在IAR5.x之前,IAR是采用的叫XLINK的链接器(它相应的配置文件为xcl文件),5.x之后才采用了新版ILINK链接器,所以咱们开发Kinetis的IAR6.x自然也采用的是ILINK链接器,配置文件为icf文件,咳咳,如果有人问什么是链接器,先谷歌一下补补,我这里就引用IAR官方手册里的一句话简要说明一下什么是链接器及其相应配置文件的作用吧:
”EWARM 5.xx 中的链接器称为ILINK。ILINK 可以从 ELF/DWARF 格式的目标文件中提取代码和数据, 并生成可执行的输出镜像。对于 ELF/DWARF 格式而言,基本的链接单元是section,section 的类型有code和data,属性可以是readonly (ro),readwrite (rw)和zeroinit (zi)。ILINK 根据 ILINK Configuration File(*.icf)来分配和定位这些sections。“
2.简单的概括icf的文件,其主要包括以下几个内容,即:
(1)可编址的存储空间(memory);
(2)不同的存储地址区域(region);
(3)不同的地址块(block);
(4)section的初始化与否;
(5)section在存储空间的放置。
上面几点内容,如果你对照实际icf文件都会找得到,建议大家尝试下,会让你受益匪浅的。
3.对于icf文件使用的常用命令,在网上早已有人贴出来了,随意即可搜到,这里省去麻烦,我也贴出来自己学的时候记下来的命令用法,建议通读一遍:
(1)define [ exported ] symbol name = expr;
作用:指定某个符号的值。
参数:
exported 导出该symbol,使其对可执行镜像可用
name 符号名
expr 符号值
举例:
define symbol RAM_START_ADDRESS = 0x40000000;  
define symbol RAM_END_ADDRESS  = 0x4000FFFF;  
-------------------------------------------------------------------
(2)define memory name with size = expr [, unit-size];
作用:
定义一个可编址的存储地址空间(memory)。
参数:
name memory的名称
expr 地址空间的大小
unit-size expr的单位,可以是位(unitbitsize),缺省是字节(unitbytesize)
举例:
define memory MEM with size = 4G;
-----------------------------------------------------------------
(3)define region name = region-expr;
作用:
定义一个存储地址区域(region)。一个区域可由一个或多个范围组成,每个范围内地址必须连续,但几个范围之间不必是连续的。
参数:
name region的名称
region-expr memory:[from expr { to expr | size expr}],可以定义起止范围,也可以定义起始地址和region的大小
举例:
define region ROM = MEM:[from 0x0 size 0x10000];
define region ROM = MEM:[from 0x0 to 0xFFFF];
---------------------------------------------------------------------------------------------
(4)
define block name [ with param, param... ]
{
extended-selectors
};
作用:   定义一个地址块(block);它可以是个只保留指定大小的地址空间的空块,比如栈、堆;也可以包含一系列的sections,由extended-selectors 选择。
参数:
name   block 的名称
param   可以是:   size = expr   (块的大小)
maximum size = expr (块大小的上限)
alignment = expr   (最小对齐字节数)
fixed order   (按照固定顺序放置sections)
extended-selector  [ first | last ] { section-selector | block name | overlay name }
first   最先存放
last   最后存放
section-selector [ section-attribute ][ section sectionname ][object filename ]
section-attribute [ readonly [ code | data ] | readwrite [ code | data ] | zeroinit ]
sectionname   section的名称
filename   目标文件的名称
name   block或overlay的名称
注:这里可以按照section的属性,名称及其所在目标文件这三个过滤条件中,任意选取一个条件或多个条件进行组合,来圈定所要求的sections。
举例:
define block HEAP with size = 0x1000, alignment = 4 { };
define block MYBLOCK1 = { section mysection1, section mysection2, readwrite };
define block MYBLOCK2 = { readwrite object myfile2.o };
define block MYBLOCK3 = { readonly code object myfile3.o };
(5)
initialize { by copy | manually } [ with param, param... ]
{
section-selectors
};
作用:   初始化sections
参数:
by copy   在程序启动时自动执行初始化
manually   在程序启动时不自动执行初始化
param   可以是:   packing = { none | compress1 | compress2 | auto } copy routine = functionname
packing表示是否压缩数据,缺省是auto
functionname表示是否使用自己的拷贝函数来取代缺省的拷贝函数
section-selector 同上
举例:
initialize by copy { readwrite };  
--------------------------------------------------------------
(6)
do not initialize
{
section-selectors
};
作用:   规定在程序启动时不需要初始化的sections;一般用于__no_init 声明的变量段(.noinit)
参数:
section-selector  同上
举例:
do not initialize { .noinit };  
(7)
place at { address memory [:expr] | start of region_expr | end of region_expr }
{
extended-selectors
};
作用:   把section 或 block 放置在某个具体的起始地址处,或者一个 region 的开始或结束处
参数:
memory   memory 的名称
expr   地址值,该地址必须在 memory 所定义的范围内
region_expr   region 的名称
extended-selector   同上
举例:
place at end of ROM { section .checksum };  
place at address MEM:0x0 { section .intvec };  
place at address MEM:0x1000 { section .text object myfile.o };  
place at address MEM:0x1000 { readonly object myfile.o };  
place at address MEM:0x1000 { readonly data object myfile.o };  
(8)
place in region-expr
{
extended-selectors
};
作用:   把section 或 block  (按任意顺序)放置在某个region 中
参数:
region-expr   region 的名称
extended-selector   同上
举例:
place in ROM { readonly };  
place in RAM { readwrite };  
place in RAM { block HEAP, block CSTACK, block IRQ_STACK };  
place in ROM { section .text object myfile.o };  
place in ROM { readonly object myfile.o };  
place in ROM { readonly data object myfile.o };  
下面为系统预定义(即你是找不到其定义的,所以不要浪费时间去找了,呵呵)的section和block描述,上图:
4.相关命令知晓了,也就是大好基础了,下面就俺就根据上面个的指令独家解析下飞思卡尔提供的Kinetis例程包里自带的icf配置文件,以512KB_Pflash.icf为例介绍一下(当初自己上传的开发框架代码里没有作相关注释,这里就算是补充了吧,哈哈):
(1)首先找到该文件,打开(咳咳,虽然这步算是废话,不过为了严谨,还是不能少的,呵呵),采用从上到下的顺序解读;
(2)

先定义了一些可读性的符号,包括异常向量表的起始地址,ROM、RAM 的起止地址和堆、栈的大小等(该地址分配我们可以在Kinetis的datasheet里找到),以前缀__ICFEDIT_开头的符号是由图形化编辑工具 ICF Editor自动定义的,可能会有些人不懂,其实上面部分代码是体现在IAR的Options->Linker选项里的(自己去探索一下即可发现)。
(3)
这部分仍然是定义一些符号,由Kinetis的内存映射可以知道,其实其内部是由两部分RAM块组成的,所以第一步出现RAM_start这一步出现了 RAM2_start,另外也定义了中断向量表在ROM中的地址和在RAM中的地址。code_start定义为0x00000410是紧邻前面向量表的,也就是说向量表占用了0x00000410大小的空间。
(4)

到了这一步就设计到具体操作内容了,32位地址总线选址4G空间,然后定义了kinetis(512k型号的哈)的ROM区的地址范围和RAM区(含 RAM1和RAM2)的地址范围。接着下面定义了堆和栈的属性,8字节对齐方式,大小为前面定义的大小即分别为0x1000和0x200。
(5)
对属性为readwrite的sections,.data和.textrw的sections不自动初始化,对.noinit属性的 sections(即用__no_init修饰的全局和静态变量),定义重定位代码区为.textrw_init,定义重定位RAM区为.textrw。
(6)
对所有的sections 和 blocks 在地址空间中所处的位置进行了配置。首先将只读的异常向量表.intvec放置在_intvec_start地址处(前面已定义),然后将余下的只读 sections以任意顺序存放在ROM_region中,将可读写的sections和栈、堆这些blocks以任意顺序存放在RAM_region 中。

阿莫论坛20周年了!感谢大家的支持与爱护!!

如果想吃一顿饺子,就得从冰箱里取出肉,剁馅儿,倒面粉、揉面、醒面,擀成皮儿,下锅……
一整个繁琐流程,就是为了出锅时那一嘴滚烫流油的热饺子。

如果这个过程,禁不住饿,零食下肚了,饺子出锅时也就不香了……《非诚勿扰3》

出0入0汤圆

发表于 2014-9-4 19:02:22 | 显示全部楼层
一般这个文件需要人为操作吗?

出0入0汤圆

发表于 2014-9-4 19:20:01 | 显示全部楼层
mark                  

出0入0汤圆

发表于 2014-9-4 19:21:33 | 显示全部楼层
这个一般不需要人为操作,但是了解这个很有帮助,知道代码在片子里空间是怎么分配的,有时候需要修改堆栈,用处就大了

出0入0汤圆

发表于 2014-9-4 19:27:34 来自手机 | 显示全部楼层
这个和xlink的xcl差不多的意思吧。

出0入0汤圆

发表于 2014-9-4 19:45:12 | 显示全部楼层
嗯,虽然目前用不着,但了解这些说不定有用啊

出100入101汤圆

发表于 2014-9-4 20:38:34 | 显示全部楼层
jzhang123 发表于 2014-9-4 19:21
这个一般不需要人为操作,但是了解这个很有帮助,知道代码在片子里空间是怎么分配的,有时候需要修改堆栈, ...

很多需要自己创建、修改。比如ADS的分散加载就需要自己写。

出0入0汤圆

发表于 2014-9-4 20:41:12 来自手机 | 显示全部楼层
mark现在用不着

出0入0汤圆

发表于 2014-9-18 08:28:08 | 显示全部楼层
MARK,以后有用到的时候!

出0入0汤圆

发表于 2014-12-4 11:54:37 | 显示全部楼层
比看map文件稍微懂一点,还得继续揣摩啊

出0入0汤圆

发表于 2014-12-4 17:54:39 | 显示全部楼层
嵌入式工程 肯定会涉及的方面  先收藏了

出0入0汤圆

发表于 2014-12-4 19:01:56 | 显示全部楼层
这个是不是应该对应KEIL的.sct文件 貌似是一样的

出0入0汤圆

发表于 2014-12-4 19:05:10 | 显示全部楼层
刚看了一下楼主转贴的地址,发现还有一些内容楼主没有转载,
以下是:
define symbol NVNC_Start = 0x08000000; // 中断起始地址
define symbol NVNC_size_cstack   = 0x400; //中断堆栈尺寸
define symbol PROEG_size_heap    = 0x400;//程序堆栈尺寸
define symbol USB_SRAM_start  = 0x40006000; //USB 专用SRAM起始与结束地址
define symbol USB_SRAM_end    = 0x400063FF;
define symbol SyS_SRAM_start   = 0x20000000;  //定议RAM起始与结束地址
define symbol SyS_SRAM_end     = 0x2000FFFF; // 64K
define symbol SyS_Flash_start   = 0x08000800 ; //定议FLASH起始与结束地址
define symbol SyS_Flash_end     = 0x0801FFFF;  //512K
//define symbol SyS_Flash_P1 = (SyS_Flash_start+(0x800*0));


define memory mem with size = 4G;

define region USB_RAM = mem:[from USB_SRAM_start to USB_SRAM_end];
define region RAM_region = mem:[from SyS_SRAM_start to SyS_SRAM_end];
define region ROM_region = mem:[from SyS_Flash_start to SyS_Flash_end];
define region SysT_RAM = mem:[from 0x0801F800 to 0x0801FFFF];

define block CSTACK with alignment = 8, size = NVNC_size_cstack { };
define block HEAP   with alignment = 8, size = PROEG_size_heap  { };

initialize by copy { readwrite };


do not initialize  { section .noinit };


place at address mem:NVNC_Start { readonly section .intvec };

place in RAM_region { readwrite,block CSTACK, block HEAP }; // 堆栈指针存放
place in ROM_region   { readonly }; // 没有定议的C文件存放在此
//定议相关文件存储空间


以上文件只参考IAR目录下的:《EWARM_DevelopmentGuide.ENU.pdf》

出0入0汤圆

发表于 2014-12-4 19:11:44 | 显示全部楼层
分享一个与.icf文件相关的网址和文件:

IAR中xcl及icf文件详解:
http://wenku.baidu.com/view/4cc8b5d528ea81c758f57849.html

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有帐号?注册

x

出0入0汤圆

发表于 2014-12-4 21:29:47 | 显示全部楼层
好东东,收藏了

出0入0汤圆

发表于 2015-3-27 22:42:15 | 显示全部楼层
IAR ILINK连接器的.ICF配置文件详解

出0入0汤圆

发表于 2017-11-27 11:40:11 | 显示全部楼层
子鱼 发表于 2014-12-4 19:11
分享一个与.icf文件相关的网址和文件:

IAR中xcl及icf文件详解:

太好了,居然能看懂一点点

出10入95汤圆

发表于 2018-4-22 19:20:22 | 显示全部楼层
IAR ICF配置文件详解
谢谢分享!

出0入0汤圆

发表于 2018-8-19 00:08:04 | 显示全部楼层
谢谢分享,最近在用realtek的sdk,采用的iar开发,里面的icf实在看不懂,看完本帖,有点理解了

出0入0汤圆

发表于 2019-2-25 16:13:04 | 显示全部楼层
早有耳闻,但一直没用过,这次对icf文件有了更深入的认识。

出0入0汤圆

发表于 2019-3-18 12:36:25 | 显示全部楼层
mark,有用

出0入0汤圆

发表于 2019-4-2 11:34:18 | 显示全部楼层
IAR确实好用,就是插件少了点

出0入0汤圆

发表于 2019-7-31 11:28:08 | 显示全部楼层
gujiamao_love 发表于 2014-9-4 19:02
一般这个文件需要人为操作吗?

在软件需要对ROM RAM检测的时候,需要小小改动下
回帖提示: 反政府言论将被立即封锁ID 在按“提交”前,请自问一下:我这样表达会给举报吗,会给自己惹麻烦吗? 另外:尽量不要使用Mark、顶等没有意义的回复。不得大量使用大字体和彩色字。【本论坛不允许直接上传手机拍摄图片,浪费大家下载带宽和论坛服务器空间,请压缩后(图片小于1兆)才上传。压缩方法可以在微信里面发给自己(不要勾选“原图),然后下载,就能得到压缩后的图片】。另外,手机版只能上传图片,要上传附件需要切换到电脑版(不需要使用电脑,手机上切换到电脑版就行,页面底部)。
您需要登录后才可以回帖 登录 | 注册

本版积分规则

手机版|Archiver|amobbs.com 阿莫电子技术论坛 ( 粤ICP备2022115958号, 版权所有:东莞阿莫电子贸易商行 创办于2004年 (公安交互式论坛备案:44190002001997 ) )

GMT+8, 2024-3-29 00:14

© Since 2004 www.amobbs.com, 原www.ourdev.cn, 原www.ouravr.com

快速回复 返回顶部 返回列表