这是在使用 Make 工具编译 glibc 时生成的。
glibc 的 Makefile 中有一条规则(由 make install 开始),它只是将需要的行回显到一些临时文件 $@.new:
(echo '/* GNU ld script';\
echo ' Use the shared library, but some functions are only in';\
echo ' the static library, so try that secondarily. */';\
cat $<; \
echo 'GROUP ( $(slibdir)/libc.so$(libc.so-version)' \
'$(libdir)/$(patsubst %,$(libtype.oS),$(libprefix)$(libc-name))'\
' AS_NEEDED (' $(slibdir)/$(rtld-installed-name) ') )' \
) > $@.new
然后这个文件被重命名为libc.so
mv -f $@.new $@
这是来自 Makefile 的评论,解释了一下:
# What we install as libc.so for programs to link against is in fact a
# link script. It contains references for the various libraries we need.
# The libc.so object is not complete since some functions are only defined
# in libc_nonshared.a.
# We need to use absolute paths since otherwise local copies (if they exist)
# of the files are taken by the linker.
我的理解是:libc.so.6 不完整,需要一些不能存储在共享库中的东西。因此,glibc 开发人员将这个东西移到了 glibc 的静态部分 - libc_nonshared.a。为了强制始终链接 libc.so.6 和 libc_nonstared.a,他们创建了一个特殊的链接脚本,指示 ld 链接器在要求 -lc (libc) 时使用两者
非共享部分是什么?让我们检查一下:
$ objdump -t /usr/lib/libc_nonshared.a |grep " F "|grep -v __
00000000 g F .text 00000058 .hidden atexit
00000000 w F .text 00000050 .hidden stat
00000000 w F .text 00000050 .hidden fstat
00000000 w F .text 00000050 .hidden lstat
00000000 g F .text 00000050 .hidden stat64
00000000 g F .text 00000050 .hidden fstat64
00000000 g F .text 00000050 .hidden lstat64
00000000 g F .text 00000050 .hidden fstatat
00000000 g F .text 00000050 .hidden fstatat64
00000000 w F .text 00000058 .hidden mknod
00000000 g F .text 00000050 .hidden mknodat
00000000 l F .text 00000001 nop
有atexit()、*stat*()、mknod 函数。为什么?真的不知道,但这是glibc的事实。
这是一些很长的解释http://giraffe-data.com/~bryanh/giraffehome/d/note/proglib,我引用了它的开头:
The stat() family of functions and mknod() are special. Their
interfaces are tied so tightly to the underlying operating system that
they change occasionally.