八、 内存区域命令

在默认情形下,连接器可以为section在程序地址空间内分配任意位置的存储区域。并通过输出section描述的> REGION属性显示地将该输出section限定于在程序地址空间内的某块存储区域,当存储区域大小不能满足要求时,连接器会报告该错误。
你也可以用MEMORY命令让在SECTIONS命令内*未*引用的selection分配在程序地址空间内的某个存储区域内。
注意:以下存储区域指的是在程序地址空间内的。
MEMORY命令的文法如下,
MEMORY {
NAME1 [(ATTR)] : ORIGIN = ORIGIN1, LENGTH = LEN1
NAME2 [(ATTR)] : ORIGIN = ORIGIN2, LENGTH = LEN2
}
NAME :存储区域的名字,这个名字可以与符号名、文件名、section名重复,因为它处于一个独立的名字空间。
ATTR :定义该存储区域的属性,在讲述SECTIONS命令时提到,当某输入section没有在SECTIONS命令内引用时,连接器会把该输入 section直接拷贝成输出section,然后将该输出section放入内存区域内。如果设置了内存区域设置了ATTR属性,那么该区域只接受满足该属性的section(怎么判断该section是否满足?输出section描述内好象没有记录该section的读写执行属性)。
ATTR属性内可以出现以下7个字符,
R 只读section
W 读/写section
X 可执行section
A ‘可分配的’section
I 初始化了的section
L 同I
! 不满足该字符之后的任何一个属性的section
ORIGIN :关键字,区域的开始地址,可简写成org或o
LENGTH :关键字,区域的大小,可简写成len或l
示例
MEMORY
{
rom (rx) : ORIGIN = 0, LENGTH = 256K
ram (!rx) : org = 0×40000000, l = 4M
}
此例中,把在SECTIONS命令内*未*引用的且具有读属性或写属性的输入section放入rom区域内,把其他未引用的输入section放入 ram。如果某输出section要被放入某内存区域内,而该输出section又没有指明ADDRESS属性,那么连接器将该输出section放在该区域内下一个能使用位置。
九、 PHDRS命令
该命令仅在产生ELF目标文件时有效。
ELF目标文件格式用program headers程序头(程序头内包含一个或多个segment程序段描述)来描述程序如何被载入内存。可以用objdump -p命令查看。
当在本地ELF系统运行ELF目标文件格式的程序时,系统加载器通过读取程序头信息以知道如何将程序加载到内存。要了解系统加载器如何解析程序头,请参考ELF ABI文档。
在连接脚本内不指定PHDRS命令时,连接器能够很好的创建程序头,但是有时需要更精确的描述程序头,那么PAHDRS命令就派上用场了。
注意:一旦在连接脚本内使用了PHDRS命令,那么连接器**仅会**创建PHDRS命令指定的信息,所以使用时须谨慎。
PHDRS命令文法如下,
PHDRS
{
NAME TYPE [ FILEHDR ] [ PHDRS ] [ AT ( ADDRESS ) ]
[ FLAGS ( FLAGS ) ] ;
}
其中FILEHDR、PHDRS、AT、FLAGS为关键字。
NAME :为程序段名,此名字可以与符号名、section名、文件名重复,因为它在一个独立的名字空间内。此名字只能在SECTIONS命令内使用。
一个程序段可以由多个‘可加载’的section组成。通过输出section描述的属性:PHDRS可以将输出section加入一个程序段,: PHDRS中的PHDRS为程序段名。在一个输出section描述内可以多次使用:PHDRS命令,也即可以将一个section加入多个程序段。
如果在一个输出section描述内指定了:PHDRS属性,那么其后的输出section描述将默认使用该属性,除非它也定义了:PHDRS属性。显然当多个输出section属于同一程序段时可简化书写。
TYPE可以是以下八种形式,
PT_NULL 0
表示未被使用的程序段
PT_LOAD 1
表示该程序段在程序运行时应该被加载
PT_DYNAMIC 
表示该程序段包含动态连接信息
PT_INTERP 3
表示该程序段内包含程序加载器的名字,在linux下常见的程序加载器是ld-linux.so.2
PT_NOTE 4
表示该程序段内包含程序的说明信息
PT_SHLIB 5
一个保留的程序头类型,没有在ELF ABI文档内定义
PT_PHDR 6
表示该程序段包含程序头信息。
EXPRESSION 表达式值
以上每个类型都对应一个数字,该表达式定义一个用户自定的程序头。
在TYPE属性后存在FILEHDR关键字,表示该段包含ELF文件头信息;存在PHDRS关键字,表示该段包含ELF程序头信息。
AT(ADDRESS)属性定义该程序段的加载位置(LMA),该属性将**覆盖**该程序段内的section的AT()属性。
默认情况下,连接器会根据该程序段包含的section的属性(什么属性?好象在输出section描述内没有看到)设置FLAGS标志,该标志用于设置程序段描述的p_flags域。
下面看一个典型的PHDRS设置
示例
PHDRS
{
headers PT_PHDR PHDRS ;
interp PT_INTERP ;
text PT_LOAD FILEHDR PHDRS ;
data PT_LOAD ;
dynamic PT_DYNAMIC ;
}
SECTIONS
{
. = SIZEOF_HEADERS;
.interp : { *(.interp) } :text :interp
.text : { *(.text) } :text
.rodata : { *(.rodata) } /* defaults to :text */
. = . + 0×1000; /* move to a new page in memory */
.data : { *(.data) } :data
.dynamic : { *(.dynamic) } :data :dynamic
}
十、版本号命令
当使用ELF目标文件格式时,连接器支持带版本号的符号。版本号也只限于ELF文件格式。
读者可以发现仅仅在共享库中,符号的版本号属性才有意义。动态加载器使用符号的版本号为应用程序选择共享库内的一个函数的特定实现版本。
可以在连接脚本内直接使用版本号命令,也可以将版本号命令实现于一个特定版本号描述文件(用连接选项–version-script指定该文件)。
该命令的文法如下,
VERSION { version-script-commands }
以下讨论用gcc
10.1. 带版本号的符号的定义(共享库内)
文件b.c内容如下,
int getVersion()
{
return 1;
}

写连接器的版本控制脚本,本例中为b.lds,内容如下
VER1.0{
getVersion;
};
VER2.0{
};

$gcc -c b.c
$gcc -shared -Wl,--version-script=b.lds -o libb.so b.o
可以在{}内填入要绑定的符号,本例中getVersion符号就与VER1.0绑定了。
那么如果有一个应用程序连接到该库的getVersion符号,那么它连接的就是VER1.0版本的getVersion符号
如果我们对b.c文件进行了升级,更改如下:
int getVersion()
{
return 101;
}

这里我对getVersion()进行了更改,其返回值的意义也进行改变,也就是它和前不兼容:
为了程序的安全,我们把b.lds更改为,
VER1.0{
};
VER2.0{
getVersion;
};

然后生成新的libb.so文件。
这时如果我们运行app.exe(它已经连接到VER1.0版本的getVersion()),就会发现该应用程序不能运行了。
提示信息如下:
./app.exe: relocation error: ./app.exe: symbol getVersion, version VER1.0 not defined in file libb.so with link time reference
因为库内没有VER1.0版本的getVersion(),只有VER2.0版本的getVersion()。
10.2、参看连接的符号的版本
对上面生成的app.exe执行以下命令:
nm app.exe | grep getVersion
结果
U new_true@@VER1.0
用nm命令发现app连接到VER1.0版本的getVersion
10.3、 GNU的扩充
在GNU中,允许在程序文件内绑定 *符号* 到 *带版本号的别名符号*
文件b.c内容如下,
int old_getVersion()
{
return 1;
}
int new_getVersion()
{
return 101;
}
__asm__(".symver old_getVersion,getVersion@VER1.0");
__asm__(".symver new_getVersion,getVersion@@VER2.0");

其中,对于VER1.0版本号的getVersion别名符号是old_getVersion;
对于VER2.0版本号的getVersion别名符号是new_getVersion,
在连接时,默认的版本号为VER2.0
供连接器用的版本控制脚本b.lds内容如下,
VER1.0{
};
VER2.0{
};
版本控制文件内必须包含版本VER1.0和版本VER2.0的定义,因为在b.c文件内有对他们的引用
再次执行以下命令编译连接b.c和app.c
gcc -c src/b.c
gcc -shared -Wl,--version-script=./lds/b.lds -o libb.so b.o
gcc -o app.exe ./src/app.c libb.so
运行:
./app.exe

结果:
Version=0x65
说明app.exe的确是连接的VER2.0的getVersion,即new_getVersion()
我们再对app.c进行修改,以使它连接的VER1.0的getVersion,即old_getVersion()
app.c文件:
#include <stdio.h>
__asm__(".symver getVersion,getVersion@VER1.0");
extern int getVersion();
int main()
{
printf("Version=%p\n", getVersion());
return 0;
}

再次编译连接b.c和app.c
运行:
./app.exe

结果:
Version=0x1
说明此次app.exe的确是连接的VER1.0的getVersion,即old_getVersion()

转载于:https://www.cnblogs.com/Zyf2016/p/6337817.html

Linux下的lds链接脚本简介(三)相关推荐

  1. linux 链接脚本,Linux下的lds链接脚本简介(一)

    每一个链接过程都由链接脚本(linker script, 一般以lds作为文件的后缀名)控制. 链接脚本主要用于规定如何把输入文件内的section放入输出文件内, 并控制输出文件内各部分在程序地址空 ...

  2. Linux下的lds链接脚本详解

    一. 概论 每一个链接过程都由链接脚本(linker script, 一般以lds作为文件的后缀名)控制. 链接脚本主要用于规定如何把输入文件内的section放入输出文件内, 并控制输出文件内各部分 ...

  3. Linux下的lds链接脚本二

    对于.lds文件,它定义了整个程序编译之后的连接过程,决定了一个可执行程序的各个段的存储位置.虽然现在我还没怎么用它,但感觉还是挺重要的,有必要了解一下. 先看一下GNU官方网站上对.lds文件形式的 ...

  4. Linux下的lds连接脚本详解,Linux链接脚本学习--lds

    许多脚本是相当简单的. 可能最简单的脚本只含有一个命令:'SECTIONS'. 你可以使用'SECTIONS'来描述输出文件的内存布局. 'SECTIONS'是一个功能很强大的命令. 假设你的程序只有 ...

  5. linux 日志压缩及清理,linux下的日志压缩脚本

    linux下的日志压缩脚本: #!/bin/bash #第一步:先定义项目列表如下: projects="project-a project-b project-c project-d&qu ...

  6. linux启停was命令,linux下的启停脚本

    linux下的根据项目名称,进行进程的启停脚本 #!/bin/bash JAVA=/usr/bin/java APP_HOME=/opt/program/qa/wechat APP_NAME=prog ...

  7. linux下lds链接脚本详解

    转载自:http://linux.chinaunix.net/techdoc/beginner/2009/08/12/1129972.shtml 一. 概论 每一个链接过程都由链接脚本(linker ...

  8. Rt-thread [三] link.lds链接脚本详解

    Rt-thread [三] link.lds文件详解 前言 程序编译流程 什么是链接 基本概念 RT-thread stm32f407 link.lds链接脚本详解 资料 前言   开一个专题,记录自 ...

  9. Linux下Apache自动监测重启脚本

    Linux下Apache自动监测重启脚本(智能化程度较高) 本站原创 [基于 署名-非商业使用-相同方式分享 2.5 协议,转载须注明链接] 本文所述apache监控脚本已经过VPS管理百科验证 由于 ...

最新文章

  1. 在CentOS 6.x上安装luajit 2.0.4
  2. HDU6964 I love counting (字典树+莫队)
  3. opencv c++ 贴图
  4. WebAssembly 浏览器中运行c/c++模块
  5. 行转列(FOR XML PATH)
  6. vue项目设置img标签的默认图片
  7. python中二维数组如何按索引找元素_按索引或坐标访问二维数组中的元素
  8. oracle 查询时间跨度一年,关于时间跨度比较的OVERLAPS函数
  9. Spring Setter依赖注入示例
  10. java中XPATH操作xml,非常便捷
  11. 【AI手机三国杀】三星Bixby闯入战场,AI功能集成颠覆人机交互
  12. 培根密码加解密(Python)
  13. android系统cpu/内存信息提取设计
  14. java web实战宝典李宁,《Java Web编程实战宝典》李宁,刘岩,张国平著【摘要 书评 在线阅读】-苏宁易购图书...
  15. 两个高斯分布乘积的理论推导
  16. ubuntu安装libaio的错误解决
  17. 菜的抠脚团队正式成立
  18. 89.破碎的玻璃横幅
  19. k8s 查看pod流量_K8s中对pod进行流量限制
  20. Fresco+Recycleview+OKhttp+Retrofit

热门文章

  1. java 内部类_Java内部类总结有哪些 没有基础该怎么学Java?
  2. vscode的IntelliCode扩展报错
  3. Tomcat如何配置X-Frame-Options头
  4. java jmx 监控_利用VisualVm和JMX远程监控Java进程
  5. 第 45 届国际大学生程序设计竞赛(ICPC)亚洲区域赛(昆明),签到题HIL
  6. 【CSP201312-3】最大的矩形,单调栈
  7. 如何制作扫描版的文档
  8. JavaScript事件函数监视
  9. RTSP/RTP/RTCP协议的区别
  10. Unity3D之NGUI基础6:UIButton按钮