编译和连接

If you change a single file, you will have to recreate the object code file from
模块间的拼合就是链接linking

Linking 链接

  • 地址和空间的分配
  • 符号解析(resolution), 符号绑定(binding)
  • 重定位 (relocation)

重定位

由于每个模块都是单独compile出来的,比如:
main.c 调用了 foo.c中的函数 foo(), 在compiler编译main.c 的时候并不知道 foo函数的地址,所以compiler暂时将目标地址搁置,等最后linking的时候,由linker去修正目标地址。地址修正的过程叫做重定位,每个需要被修正的地方叫做重定位入口(relocation entry)。

链接与符号

在链接中,将函数变量统称为符号 (symbol)。
每个object file中都会有一个相应的符号表 (symbol table),这个表中记录了目标文件中用到的所有符号。
符号表中最重要的两类符号:

  • 本目标文件中的全局符号,可以被其他目标文件引用
  • 本目标文件中引用的全局符号,却没有定义在本目标文件中,这类是外部符号,如printf
#include <iostream>
using namespace std;

int g_var1 = 100;
int g_var2;

void foo(int i)
{
    cout << "i = " << i << endl;
}

int main()
{
    static int s_var1 = 99;
    static int s_var2;
    int a = 1;
    int b;
    foo(g_var1 +s_var1 + a);
}

针对上面这个C++代码,可以通过nm命令查看其符号表。

nm object文件名

C++ 符号修饰

符号修饰(decorate)也称为符号改编(mangle), compiler在将C++源码编译成object file时,会将函数和变量的名字进行修饰,形成符号名。

gcc编译器的symbol decoration

修饰方法

  • 所有的符号都以_Z开头
  • 对于嵌套的名字(在namespace或者class里),后面紧跟N
  • 每个名字前面是名字长度
  • E结尾
  • 对于函数,参数表紧跟在E后面,对于int类型就是一个小写i

当查看符号表的时候想查看装饰前的名字的时候,推荐用如下命令:


demangle

extern "C": 将C++ code处理成C code

extern "C"用法举例

可以看到extern "C"包裹的变量就没有经过修饰了

宏 __cplusplus

头文件声明了C语言的函数或变量,可以被C code或者C++ code引用。
如 C语言函数库string.h中的

void* memset(void*, int, size_t);
  • C code可以正确引用 memset
  • C++ code会修饰 memset, 这将导致linker无法与函数库string.h中的memset符号进行链接

C语言不支持extern "C"语法,如果为了兼容C语言和C++定义两套头文件,未免过于繁琐。

解决问题的方法是利用C++的宏 __cplusplus, C++编译器会在编译C++ code的时候默认定义这个宏,可以使用条件宏来判断当前编译单元是不是C++ code。

使用条件宏来判断当前编译单元是不是C++ code

如果正在编译C++ code,则memsetextern "C"包裹,否则,就是直接声明。上面这段代码中的技巧在所有的系统头文件里都被用到。

目标文件

  • 可执行文件:
  1. Linux下位ELF (executable linkable format)
  2. Windows下PE (portable executable)
  • 查看文件格式的命令:file
file filename
查看可执行文件

目标文件的构成

C code and object file

objdump: Display information form object file

use objdump to display info from object

size和offset

object file在ELF中的结构

可以用size命令查看ELF文件的代码段、数据段和bss段的长度
用size命令查看ELF文件的代码段、数据段和bss段的长度

readelf: Display information about ELF files

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 211,561评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,218评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,162评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,470评论 1 283
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,550评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,806评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,951评论 3 407
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,712评论 0 266
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,166评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,510评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,643评论 1 340
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,306评论 4 330
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,930评论 3 313
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,745评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,983评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,351评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,509评论 2 348

推荐阅读更多精彩内容