通过GCC Specs修改编译器的默认行为

Posted by 云谷计算 on May 2, 2018

最近希望改一下gcc的默认行为,默认能不要优化frame-pointer, 10年前做过类似的事情,当时在ChinaUnix博客上记录了下,居然被我找到了。


什么是gcc spec

其实简单来说,gcc spec文件是用来控制gcc的默认行为的,一般被放在这个目录下可以找到这个文件:

# cd /x86toolchain/lib/gcc/i686-mot-linux-gnu/3.4.3
# ls specs

也可以通过这个命令来打印gcc spec:

# gcc -dumpspecs

如果希望使用自己的specs, 可以通过-specs参数来指定:

# g++ -O2 -specs=/tmp/specs 1.c

下面是用来修改gcc默认行为的几个例子:

-O2 & -fomit-frame-pointer

默认情况下, 对于X86 GCC, 如果指定-O2, frame pointer并不会被omit掉,通过修改specs, 我们可以在-O2的情况下,同时omit frame pointer:

*cc1_options:
%{pg:%{fomit-frame-pointer:%e-pg and -fomit-frame-pointer are incompatible}} %1 %{!Q:-quiet} -dumpbase %B %{d*} %{m*} %{a*} %{c|S:%{o*:-auxbase-strip %*}%{!o*:-auxbase %b}}%{!c:%{!S:-auxbase %b}} %{g*} %{O*} %{W*&pedantic*} %{w} %{std*} %{ansi} %{v:-version} %{pg:-p} %{p} %{f*} %{undef} %{Qn:-fno-ident} %{--help:--help} %{--target-help:--target-help} %{!fsyntax-only:%{S:%W{o*}%{!o*:-o %b.s}}} %{fsyntax-only:-o %j} %{-param*} %{O2:-O2 -fomit-frame-pointer} %{O3:-O3 -fomit-frame-pointer}
*cc1plus_options:
%{O2:-O2 -fomit-frame-pointer} %{O3:-O3 -fomit-frame-pointer}

-g

默认情况下, GCC不会在编译object的时候使用-g选项,可以通过修改specs来对所有的objects在编译的时候加上-g选项:

*cc1:
%(cc1_cpu) %{profile:-p} -g

*cc1plus:
-g

后记

10年时间里,gcc已经发生了很多变化:

  1. 好像从4.6开始,gcc -O默认优化frame pointer, 倾向使用利用ELF的dwarf或者lbr(PMC机制)获取调用栈, perf record的–call-graph现在也支持这两个参数。
  2. CentOS 7的gcc已经没有默认的specs文件,可以利用strace找到gcc默认的 specs路径,比如从下面的输出可以看出,specs文件为/usr/lib/gcc/x86_64-redhat-linux/4.8.5/specs。
strace gcc -v
...
access("/usr/lib/gcc/x86_64-redhat-linux/4.8.5/", X_OK) = 0
access("/usr/lib/gcc/x86_64-redhat-linux/4.8.5/", X_OK) = 0
access("/usr/lib/gcc/x86_64-redhat-linux/4.8.5/specs", R_OK) = -1 ENOENT (No such file or directory)
access("/usr/lib/gcc/x86_64-redhat-linux/4.8.5/../../../../x86_64-redhat-linux/lib/x86_64-redhat-linux/4.8.5/specs", R_OK) = -1 ENOENT (No such file or directory)
access("/usr/lib/gcc/x86_64-redhat-linux/4.8.5/../../../../x86_64-redhat-linux/lib/specs", R_OK) = -1 ENOENT (No such file or directory)
write(2, "Using built-in specs.\n", 22Using built-in specs.
) = 22
...

参考

  1. http://www.mingw.org/wiki/SpecsFileHOWTO
  2. http://www.brendangregg.com/perf.html#StackTraces