以前了解过ld在链接的时候,是以目标文件(.o)为单位的,所以libc里的源文件基本上都很短,甚至只定义一个函数。
为的是在链接的时候,不要把没有用到的函数也链进去,从而减小可执行文件或者动态库的大小。
(1) 但是如果你将很多函数都定义在一个源文件中了,又想要减size,怎么办呢?
Stack overflow上有一篇文章
C/C++ gcc & ld - remove unused symbols (
http://stackoverflow.com/questions/6687630/c-c-gcc-ld-remove-unused-symbols)
有讨论这个问题,里面提到的解决办法是编译每个源文件时加上编译选项
-
fdata
-
sections
-
ffunction
-
sections, 链接时加上选项-Wl,--gc-sections
这个例子就能很好地验证上述方法:
#include <stdio.h>
void deadcode() { printf("This is d dead codez\n"); }
int main(void) { printf("This is main\n"); return 0 ; }
gcc -Os test.c -o test.elf
gcc -Os -fdata-sections -ffunction-sections test.c -o test.elf -Wl,--gc-sections
gcc -Os -fdata-sections -ffunction-sections test.c -o test.elf -Wl,--gc-sections -Wl,--strip-all
比较一下test.elf的大小吧。
(2) 我写这篇文章的起因其实不是为了去除某个目标文件中没有用到的函数,而是想在链接时去除掉没有用到的目标文件。
背景是我在实现某个feature时,一开始用了一种方法,后来又用了另一种方法,两种方法都有几个源文件。第一种方法由于某种原因生成的目标文件很大,而第二种方法则可以大幅减小size。
在写Makefile的时候呢,由于项目比较小,没有精确地去控制链接哪些文件,所以导致一股脑地把源文件都编译了,把目标文件都加到链接命令里了。
结果当然是可执行文件的大小没有减下去了。
解决办法很简单,只把需要的目标文件链接进去就可以了,这样size就能减小了。
#---------------------------------------------------------------------------------
#OBJS := $(patsubst src/%.cpp,obj/%.o,$(wildcard src/*.cpp))
#OBJS += $(patsubst src/%.c,obj/%.o,$(wildcard src/*.c))
EXCLUDES := $(shell cat ./exclude.txt) # put all files you want to exclude into this file
CPPFILES := $(shell find src/ -name '*.cpp')
NEWCPPFILES := $(filter-out $(EXCLUDES), $(CPPFILES))
CFILES := $(shell find src/ -name '*.c')
NEWCFILES := $(filter-out $(EXCLUDES), $(CFILES))
OBJS := $(patsubst src/%.cpp,obj/%.o,$(NEWCPPFILES))
OBJS += $(patsubst src/%.c,obj/%.o,$(NEWCFILES))
#---------------------------------------------------------------------------------
(debug makefile:
1. refer to: coolshell.cn/articles/3790.html
2. remake --trace
#---------------------------------------------------------------------------------
#OBJS := $(patsubst src/%.cpp,obj/%.o,$(wildcard src/*.cpp))
#OBJS += $(patsubst src/%.c,obj/%.o,$(wildcard src/*.c))
EXCLUDES := $(shell cat ./exclude.txt) # put all files you want to exclude into this file
CPPFILES := $(shell find src/ -name '*.cpp')
NEWCPPFILES := $(filter-out $(EXCLUDES), $(CPPFILES))
CFILES := $(shell find src/ -name '*.c')
NEWCFILES := $(filter-out $(EXCLUDES), $(CFILES))
OBJS := $(patsubst src/%.cpp,obj/%.o,$(NEWCPPFILES))
OBJS += $(patsubst src/%.c,obj/%.o,$(NEWCFILES))
#---------------------------------------------------------------------------------
(debug makefile:
1. refer to: coolshell.cn/articles/3790.html
2. remake --trace
办法很简单,但我觉得这个和我的理解有点不一样,所以就google了一下。
顺便说一下,这个blog里的关于linker的一系列文章写的很棒。