6.9. Glibc-2.28

Glibc 软件包包含主要的 C 语言库。它提供用于分配内存、检索目录、 打开和关闭文件、读写文件、字符串处理、模式匹配、 算术等用途的基本子程序。

估计编译时间: 24 SBU
需要硬盘空间: 2.8 GB

6.9.1. 安装 Glibc

[注意]

注意

Glibc 构建系统是自给自足的, 即使编译器 specs 文件和链接器仍然指向 /tools, 也能完美地安装 Glibc 。在安装 Glibc 之前不能调整工具链, 否则 Glibc 的 autoconf 测试会给出错误结果, 结果无法达成干净地构建 Glibc 这一目的。

某些 Glibc 程序使用与 FHS 不兼容的 /var/db 目录存放运行时数据。应用下列补丁, 使得这些程序在 FHS 兼容的位置存储运行时数据:

patch -Np1 -i ../glibc-2.28-fhs-1.patch

创建一个与最终的 LFS 系统兼容的符号链接, 避免最终的 Glibc 包含对 /tools 的引用:

ln -sfv /tools/lib/gcc /usr/lib

判定 GCC 包含文件目录,并为了兼容 LSB 创建一个符号链接。另外, 对于 x86_64 ,还要创建一个动态链接器的兼容性符号链接, 使它能够正常工作:

case $(uname -m) in
    i?86)    GCC_INCDIR=/usr/lib/gcc/$(uname -m)-pc-linux-gnu/8.2.0/include
            ln -sfv ld-linux.so.2 /lib/ld-lsb.so.3
    ;;
    x86_64) GCC_INCDIR=/usr/lib/gcc/x86_64-pc-linux-gnu/8.2.0/include
            ln -sfv ../lib/ld-linux-x86-64.so.2 /lib64
            ln -sfv ../lib/ld-linux-x86-64.so.2 /lib64/ld-lsb-x86-64.so.3
    ;;
esac

删除上次编译时可能留下的文件:

[注意]

译注

译者的惨痛教训表明,如果存在这个文件, 构建系统会无限循环。

rm -f /usr/include/limits.h

Glibc 文档推荐在专用目录中构建它:

mkdir -v build
cd       build

准备安装 Glibc:

CC="gcc -isystem $GCC_INCDIR -isystem /usr/include" \
../configure --prefix=/usr                          \
             --disable-werror                       \
             --enable-kernel=3.2                    \
             --enable-stack-protector=strong        \
             libc_cv_slibdir=/lib
unset GCC_INCDIR

新的配置选项和参数的含义:

CC="gcc -isystem $GCC_INCDIR -isystem /usr/include"

设定 GCC 和系统的包含文件目录,避免调试符号中包含无效路径。

--disable-werror

该选项禁用 GCC 的 -Werror 选项, 这对于运行测试套件来说是必须的。

--enable-stack-protector=strong

该选项通过加入额外代码, 对栈溢出攻击等导致的缓冲区溢出进行检查,以提高系统安全性。

libc_cv_slibdir=/lib

这个变量纠正库文件安装位置, 我们不希望使用 lib64 目录。

编译该软件包:

make
[重要]

重要

在本节中, Glibc 的测试套件十分关键,在任何情况下都不能跳过。

通常来说,可能会有极少数测试不能通过, 下面列出的失败结果一般可以安全地忽略。执行以下命令进行测试:

make check

您可能看到一些失败结果。 Glibc 的测试套件和宿主系统之间有某种依赖关系,以下是在一些版本的 LFS 上最常见的问题:

  • 已知 misc/tst-ttyname 在 LFS chroot 环境中会失败。

  • 已知 inet/tst-idna_name_classify 在 LFS chroot 环境中会失败。

  • 已知 posix/tst-getaddrinfo4posix/tst-getaddrinfo5 在某些硬件架构上会失败。

  • 已知 nss/tst-nss-files-hosts-multi 可能失败,原因尚未查明。

  • 如果 CPU 不是较新的 Intel 或 AMD 处理器, 数学测试有时会失败。

在安装 Glibc 时,它会抱怨文件 /etc/ld.so.conf 不存在。尽管这是一条无害的消息,执行以下命令即可防止这个警告:

touch /etc/ld.so.conf

修正生成的 Makefile , 跳过一个在 LFS 的不完整环境中会失败的完整性检查:

sed '/test-installation/s@$(PERL)@echo not running@' -i ../Makefile

安装该软件包:

make install

安装 nscd 的配置文件和运行时目录:

cp -v ../nscd/nscd.conf /etc/nscd.conf
mkdir -pv /var/cache/nscd

下面,安装一些 locale ,它们可以使得系统用不同语言响应用户请求。 这些 locale 都不是必须的,但是如果缺少了它们中的某些, 在将来运行软件包的测试套件时,可能跳过重要的测试。

可以用 localedef 程序安装单独的 locale 。 例如,下面的第一个 localedef 命令将 /usr/share/i18n/locales/cs_CZ 中的字符集无关 locale 定义和 /usr/share/i18n/charmaps/UTF-8.gz 中的字符映射定义组合起来,并附加到 /usr/lib/locale/locale-archive 文件。 以下命令将会安装能够覆盖测试所需的最小 locale 集合:

mkdir -pv /usr/lib/locale
localedef -i cs_CZ -f UTF-8 cs_CZ.UTF-8
localedef -i de_DE -f ISO-8859-1 de_DE
localedef -i de_DE@euro -f ISO-8859-15 de_DE@euro
localedef -i de_DE -f UTF-8 de_DE.UTF-8
localedef -i en_GB -f UTF-8 en_GB.UTF-8
localedef -i en_HK -f ISO-8859-1 en_HK
localedef -i en_PH -f ISO-8859-1 en_PH
localedef -i en_US -f ISO-8859-1 en_US
localedef -i en_US -f UTF-8 en_US.UTF-8
localedef -i es_MX -f ISO-8859-1 es_MX
localedef -i fa_IR -f UTF-8 fa_IR
localedef -i fr_FR -f ISO-8859-1 fr_FR
localedef -i fr_FR@euro -f ISO-8859-15 fr_FR@euro
localedef -i fr_FR -f UTF-8 fr_FR.UTF-8
localedef -i it_IT -f ISO-8859-1 it_IT
localedef -i it_IT -f UTF-8 it_IT.UTF-8
localedef -i ja_JP -f EUC-JP ja_JP
localedef -i ru_RU -f KOI8-R ru_RU.KOI8-R
localedef -i ru_RU -f UTF-8 ru_RU.UTF-8
localedef -i tr_TR -f UTF-8 tr_TR.UTF-8
localedef -i zh_CN -f GB18030 zh_CN.GB18030

另外,安装适合您自己国家、语言和字符集的 locale 。

[注意]

译注

建议中文用户安装 zh_CN.UTF-8 作为日常使用的 locale。GB18030 是为了后向兼容古老的 GB2312 而设计的编码, 在现代 Linux 系统和互联网使用时会引起一些奇怪的问题。 中华人民共和国国家标准化管理委员会的官方网站已经切换到 UTF-8。

或者,也可以一次安装 glibc-2.28/localedata/SUPPORTED 中列出的所有 locale (包括上面列出的所有 locale,以及其他很多)。 执行下面这个需要很长时间的命令:

make localedata/install-locales

如果需要,再使用 localedef 命令创建和安装 glibc-2.28/localedata/SUPPORTED 中没有列出的 locale ,当然您不太可能需要它们。

[注意]

注意

目前 glibc 在解析国际化域名时使用 libidn2, 形成了一个运行时依赖关系。如果需要使用解析国际化域名的功能,参阅 BLFS libidn2 页面 安装 libidn2。

6.9.2. 配置 Glibc

6.9.2.1. 添加 nsswitch.conf

由于 Glibc 的默认值在网络环境下不能很好地工作, 需要创建配置文件 /etc/nsswitch.conf

执行以下命令创建新的 /etc/nsswitch.conf

cat > /etc/nsswitch.conf << "EOF"
# Begin /etc/nsswitch.conf

passwd: files
group: files
shadow: files

hosts: files dns
networks: files

protocols: files
services: files
ethers: files
rpc: files

# End /etc/nsswitch.conf
EOF

6.9.2.2. 添加时区数据

输入以下命令,安装并设置时区数据:

tar -xf ../../tzdata2018e.tar.gz

ZONEINFO=/usr/share/zoneinfo
mkdir -pv $ZONEINFO/{posix,right}

for tz in etcetera southamerica northamerica europe africa antarctica  \
          asia australasia backward pacificnew systemv; do
    zic -L /dev/null   -d $ZONEINFO       -y "sh yearistype.sh" ${tz}
    zic -L /dev/null   -d $ZONEINFO/posix -y "sh yearistype.sh" ${tz}
    zic -L leapseconds -d $ZONEINFO/right -y "sh yearistype.sh" ${tz}
done

cp -v zone.tab zone1970.tab iso3166.tab $ZONEINFO
zic -d $ZONEINFO -p America/New_York
unset ZONEINFO

zic 命令的含义:

zic -L /dev/null ...

该命令创建没有闰秒的 POSIX 时区。一般的惯例是将它们安装在 zoneinfozoneinfo/posix 两个目录中。前者是必须的,否则若干测试套件会报告错误。 在嵌入式系统上,如果存储空间十分紧张, 而且您永远不会更新时区信息,您可以不使用 posix 目录, 以节约 1.9 MB,但个别程序或测试套件可能会失败。

zic -L leapseconds ...

该命令创建正确的,包含闰秒的时区。在嵌入式系统上, 如果存储空间十分紧张,而且您永远不会更新时区信息, 也不关心系统时间是否正确,您可以跳过 right 目录, 以节约 1.9 MB。

zic ... -p ...

该命令创建 posixrule 文件。 我们使用纽约时区,因为 POSIX 要求与美国一致的夏令时规则。

一种确定本地时区的方法是运行脚本:

tzselect

在回答关于当前位置的若干问题后,脚本会输出对应时区的名字 (例如America/Edmonton)。 在 /usr/share/zoneinfo 中还有一些该脚本不能识别,但可以使用的时区,如 Canada/Eastern 或者 EST5EDT

确定时区后,执行以下命令,创建 /etc/localtime

cp -v /usr/share/zoneinfo/<xxx> /etc/localtime

<xxx> 替换成选定时区的名称 (例如 Canada/Eastern)。

6.9.2.3. 配置动态加载器

默认情况下,动态加载器 (/lib/ld-linux.so.2) 在 /lib/usr/lib 中搜索程序运行时需要的动态库。然而,如果在其他目录中有动态库, 为了使动态加载器能够找到它们,需要把这些目录添加到文件 /etc/ld.so.conf 中。 有两个目录 /usr/local/lib/opt/lib 经常包含附加的共享库,所以现在将它们添加到动态加载器的搜索目录中。

运行以下命令,创建一个新的 /etc/ld.so.conf

cat > /etc/ld.so.conf << "EOF"
# Begin /etc/ld.so.conf
/usr/local/lib
/opt/lib

EOF

如果希望的话,动态加载器也可以搜索一个目录,并将其中的文件包含在 ld.so.conf 中。 通常包含文件目录中的文件只有一行,指定一个期望的库文件目录。 如果需要这项功能,执行以下命令:

cat >> /etc/ld.so.conf << "EOF"
# Add an include directory
include /etc/ld.so.conf.d/*.conf

EOF
mkdir -pv /etc/ld.so.conf.d

6.9.3. Glibc 的内容

安装的程序: catchsegv, gencat, getconf, getent, iconv, iconvconfig, ldconfig, ldd, lddlibc4, locale, localedef, makedb, mtrace, nscd, pldd, sln, sotruss, sprof, tzselect, xtrace, zdump, 以及 zic
安装的库: ld-2.28.so, libBrokenLocale.{a,so}, libSegFault.so, libanl.{a,so}, libc.{a,so}, libc_nonshared.a, libcidn.so, libcrypt.{a,so}, libdl.{a,so}, libg.a, libieee.a, libm.{a,so}, libmcheck.a, libmemusage.so, libnsl.{a,so}, libnss_compat.so, libnss_dns.so, libnss_files.so, libnss_hesiod.so, libnss_nis.so, libnss_nisplus.so, libpthread.{a,so}, libpthread_nonshared.a, libresolv.{a,so}, librpcsvc.a, librt.{a,so}, libthread_db.so, 以及 libutil.{a,so}
安装的目录: /usr/include/arpa, /usr/include/bits, /usr/include/gnu, /usr/include/net, /usr/include/netash, /usr/include/netatalk, /usr/include/netax25, /usr/include/neteconet, /usr/include/netinet, /usr/include/netipx, /usr/include/netiucv, /usr/include/netpacket, /usr/include/netrom, /usr/include/netrose, /usr/include/nfs, /usr/include/protocols, /usr/include/rpc, /usr/include/rpcsvc, /usr/include/sys, /usr/lib/audit, /usr/lib/gconv, /usr/lib/locale, /usr/libexec/getconf, /usr/share/i18n, /usr/share/zoneinfo, /var/cache/nscd, 以及 /var/lib/nss_db

简要描述

catchsegv

在程序因为段错误而终止时创建栈跟踪

gencat

生成消息目录

getconf

显示文件系统指定的系统配置变量值

getent

从管理数据库取得条目

iconv

转换给定文件的编码

iconvconfig

创建快速装入 iconv 模块配置文件

ldconfig

设置运行时动态链接

ldd

报告给定程序或共享库依赖于哪些共享库

lddlibc4

辅助 ldd 处理对象文件

locale

给出当前区域的一些信息

localedef

编译 locale 规范

makedb

从文本输入创建简单的数据库

mtrace

读取并解析内存跟踪文件,以人类可读的形式输出内存跟踪信息

nscd

一个缓存最常见命名服务请求的守护进程

pldd

列出正在运行的进程使用的共享库

sln

静态链接的 ln 程序

sotruss

跟踪特定命令对共享库中子程序的调用

sprof

读取并显示共享库性能剖析数据

tzselect

询问用户系统所在的位置并报告对应的时区

xtrace

显示正在执行的函数以跟踪程序执行

zdump

输出当前时间在多个时区中的表示

zic

时区编译器

ld-2.28.so

动态链接器/加载器

libBrokenLocale

被 Glibc 内部用作使某些不正确的程序(例如某些 Motif 程序) 正常运行的粗糙手段,参阅 glibc-2.28/locale/broken_cur_max.c 中的注释了解更多信息

libSegFault

catchsegv 使用的段错误信号处理程序

libanl

异步的命名查找库

libc

主要的 C 运行库

libcidn

被 Glibc 内部的 getaddrinfo 函数 用于处理国际化域名

libcrypt

密码学库

libdl

动态链接接口库

libg

没有功能的空库,曾经是 g++ 的运行库。

libieee

链接该库以使用国际电气电子工程师学会(IEEE) 定义的数学函数错误处理规则,而不是默认的 POSIX.1 错误处理。

libm

数学库

libmcheck

链接到该库时启用内存分配检查

libmemusage

memusage 用于收集程序内存使用信息

libnsl

网络服务库

libnss

命名服务开关库,包含用于解析域名、用户名、组名、代号、 服务、协议等的函数。

libpthread

POSIX 线程库

libresolv

包含用于创建、发送和解析因特网域名服务数据包的函数。

librpcsvc

包含一些 RPC 服务函数

librt

包含 POSIX.1b 实时扩展要求的多数接口

libthread_db

包含用于构建多线程程序调试的函数器

libutil

包含许多 Unix 工具使用的 标准 函数