第 3.5 节 通过源代码 ports 方式安装软件

注意

ports 和 pkg 可以同时使用,而且大部分人也是这么用的。但是要注意 pkg 的源必须是 latest,否则会存在一些依赖上比如 lib 库上的问题。latest 的源也比主线上的 ports 要出来的晚(是从中编译出来的),因此即使是 latset 源也可能会出现上述问题,总之有问题出现时就卸载那个 pkg 安装的包,重新使用 ports 编译即可。

FreeBSD ports 基本用法

警告

FreeBSD 关于在未来弃用 portsnap 的说明:[HEADS UP] Planned deprecation of portsnap

以下是邮件翻译:

我们正计划废除在 ports 中使用 portsnap 的做法。

原因如下 (无特定排序)。

  • Portsnap 不支持季度分支,即使在季度分支被创建并改为非 HEAD 软件包的默认值的多年之后。
  • 与 svn 或 git 相比,Portsnap 似乎并不节省磁盘空间,如果你算上元数据(存储的)的话。算上元数据(默认存储在 /var/db/portsnap 中),并且你对 svn 或 git 做一个没有历史记录的相同的比较,并且忽略可能的 ZFS 压缩。也就是说,你用 svn exportgit clone --depth 1,你会看到这样的磁盘用量:
    342M svnexport
    426M git
    477M portsnap
  • Portsnap 也不像 git 那样可以离线工作。使用 git,你可以也可以通过运行 git pull --unshallow 轻松添加历史记录。
  • 这种从 portsnap 的迁移与计划中的向 git 的迁移很相称。
  • 另外,根据我们在 Bugzilla 上看到的补丁,使用使用 portsnap 导致人们很容易意外地提交补丁到 Bugzilla,而这些补丁并不容易应用。
  • 由于 portsnap 不支持季度分支,它经常导致用户在错误的分支上进行编译,或最终使用不匹配的软件包。也就是说,他们通过 pkg 从季度分支安装软件包,然后想要定制,因此运行 portsnap 并从 head 编译,这可能会导致问题。正如我们经常看到的那样。即使这种情况没有发生,也会增加故障排除的几率,以确认它没有发生。

我们知道人们已经习惯了 portsnap,但我们相信:

  • 人们应该能够轻松地使用在基本系统或 git 来使用 pkg 的 svnlite(似乎很少有人真正使用 `WITHOUT_SVNLITE``)。
  • 也有可能退回到来获取 tar 或 zip。从 https://cgit-beta.freebsd.org/ports/,尽管这确实使更新难度增加。

我们将如何做,按顺序进行:

  • 更新 poudriere 以默认使用 svn。这已经完成了:

https://github.com/freebsd/poudriere/pull/764

https://github.com/freebsd/poudriere/commit/bd68f30654e2a8e965fbdc09aad238c8bf5cdc10

我们欢迎任何有建设性的反馈。所有的意见都会被听取,如果计划需要修改,我们会在几周内把修改后的计划反馈给你。这个过程将需要一些时间,但希望不会对任何人的正常工作流程造成太大的干扰。

Steve (portmgr@)

首先获取 ports

请参照换源章节。

使用 whereis 查询软件路径

# whereis python

输出 python: /usr/ports/lang/python

如何安装 python3:

# cd /usr/ports/lang/python
# make BATCH=yes clean

其中 BATCH=yes 的意思是使用默认配置

加速编译

FreeBSD ports 多线程编译(推荐)

  • FreeBSD ports 多线程编译
FORCE_MAKE_JOBS=yes
MAKE_JOBS_NUMBER=4

Linux 如 Gentoo 上一般是直接 -jx 或者 jx+1, x 为核心数。

写入 /etc/make.conf 没有就 touch 新建一个。

4 是处理器核心数(还是线程数?),不知道就别改。英特尔的处理器搜索 CPU型号+ARK 转跳英特尔官网可查询线程数。

  • 个别情况下可以设置别名加速编译:(非永久设置,FreeBSD 14 无需设置,自动生效)
# alias ninja='ninja -j4'

设置内存为 tmp

ee /etc/fstab 写入:

tmpfs /tmp tmpfs rw 0 0

重启。

参考资料:tmpfs -- in-memory file system

ccache

这部分现在有点问题,先不要用,见 Bug

警告

使用 ccache 可能会导致编译失败!只在重复编译时起效果,首次编译不仅不会加速还会慢上一些。是一种以空间换时间的行为。

目前最新版本是 ccache4:

# pkg install ccache4

# cd /usr/ports/devel/ccache4/ && make install clean
  • ksh/sh 添加到 /etc/profile:
export CCACHE_PREFIX=distcc
export DISTCC_HOSTS="localhost host1 host2"
  • csh/tcsh 添加到 /etc/csh.cshrc:
setenv CCACHE_PREFIX distcc
setenv DISTCC_HOSTS "localhost host1 host2"

做软链接:

# ccache-update-links
root@ykla:/usr/ports/devel/ccache4 # ccache -M 5G  # 设置编译缓存最大为 5GB
Set cache size limit to 5.0 GB

root@ykla:/usr/ports/devel/ccache4 # ccache -s  # 查看编译缓存
Local storage:
  Cache size (GB): 0.0 / 5.0 ( 0.00%)

root@ykla:~ # find /  -name ccache.conf # 全局查找配置文件路径
/root/.config/ccache/ccache.conf

参考文献:

如何使用多线程下载:

  • axel
# pkg install axel

新建或者编辑 # ee /etc/make.conf 文件,写入以下几行:

FETCH_CMD=axel
FETCH_BEFORE_ARGS= -n 10 -a
FETCH_AFTER_ARGS=
DISABLE_SIZE=yes
  • wget2

wget2 正在进行测试!先不要用

# cd /usr/ports/www/wget2/ && make install clean

新建或者编辑 # ee /etc/make.conf 文件,写入以下几行:

FETCH_CMD=wwget2
FETCH_BEFORE_ARGS= -c -t 3 -o 10
FETCH_AFTER_ARGS=
DISABLE_SIZE=yes

-c 断点续传;-t 3 重试次数 3; -o 10 启用 10 个线程进行下载。

10 这个参数可能过于保守,我一般直接用 50 或 100。但是要注意很多服务器不支持这么多线程同时下载。

进阶

如果不选择 BATCH=yes 的方法手动配置:

看看 python 的 ports 在哪:

# whereis python
# python: /usr/ports/lang/python

安装 python3:

# cd /usr/ports/lang/python

如何设置全部所需的依赖:

# make config-recursive

如何删除当前 port 及其依赖的配置文件:

# make rmconfig-recursive

如何一次性下载所有需要的软件包:

# make BATCH=yes fetch-recursive

升级 ports

# portsnap auto

ports 编译的软件也可以转换为 pkg 包

# pkg create nginx

FreeBSD 包升级管理工具

首先更新 Ports

# portsnap auto

然后列出过时 Ports 组件

# pkg_version -l '<'
7-zip-22.01                        <
AppStream-0.16.1                   <
alsa-plugins-1.2.2_11              <
aom-3.7.0.r1                       <
ark-23.04.3                        <
baloo-widgets-23.04.3              <
brotli-1.0.9,1                     <
ca_root_nss-3.92                   <
cargo-c-0.9.23                     <
chromium-116.0.5845.110            <
consolekit2-1.2.6_1                <
discount-2.2.7c                    <
dolphin-23.04.3                    <
…………

下边分别列出 2 种 FreeBSD 手册中提及的升级工具:

一、portupgrade

# cd /usr/ports/ports-mgmt/portupgrade && make install clean
# portupgrade -ai #自动升级所有软件
# portupgrade -R screen #升级单个软件

二、portmaster (推荐)

  • 更新:
# cd /usr/ports/ports-mgmt/portmaster && make install clean
# portmaster -a #自动升级所有软件
# portmaster screen #升级单个软件
  • 查看依赖关系:
root@ykla:/usr/ports/ports-mgmt/portmaster # portmaster sysutils/htop  --show-work

===>>> Port directory: /usr/ports/sysutils/htop

===>>> Starting check for all dependencies
===>>> Gathering dependency list for sysutils/htop from ports

===>>> Installed devel/autoconf
===>>> Installed devel/automake
===>>> NOT INSTALLED		devel/libtool
===>>> NOT INSTALLED		devel/pkgconf
===>>> NOT INSTALLED		lang/python311
===>>> Installed ports-mgmt/pkg
  • 安装软件

参考资料:

FreeBSD USE

  • 如何指定 Ports 编译的版本?

如 Python 现在的默认编译版本是 3.9,要改为 3.11,:

# echo "DEFAULT_VERSIONS+= python=3.11  python3=3.11" >> /etc/make.conf

如果只设置了单个参数,那么出现警告是正常的,见 Bug

/!\ WARNING /!\

PYTHON_DEFAULT must be a version present in PYTHON2_DEFAULT or PYTHON3_DEFAULT,
if you want more Python flavors, set BUILD_ALL_PYTHON_FLAVORS in your make.conf

完整的列表见 https://cgit.freebsd.org/ports/tree/Mk/bsd.default-versions.mk

参考资料:

# echo "OPTION_UNSET+= MYSQL" >> /etc/make.conf

查看依赖

已经安装:

root@ykla:~ # pkg info -d screen
screen-4.9.0_6:
	indexinfo-0.3.1

未安装:

root@ykla:/usr/ports/sysutils/htop # make all-depends-list
/usr/ports/ports-mgmt/pkg
/usr/ports/devel/pkgconf
/usr/ports/devel/kyua
/usr/ports/devel/lutok
/usr/ports/lang/lua54
/usr/ports/devel/libedit
/usr/ports/databases/sqlite3
/usr/ports/lang/tcl86
/usr/ports/lang/python311
/usr/ports/devel/gettext-runtime
/usr/ports/print/indexinfo
/usr/ports/devel/gettext-tools
/usr/ports/devel/libtextstyle
/usr/ports/devel/libffi
/usr/ports/misc/dejagnu
/usr/ports/devel/gmake
/usr/ports/lang/expect
/usr/ports/devel/autoconf
/usr/ports/devel/m4
/usr/ports/print/texinfo
/usr/ports/misc/help2man
/usr/ports/devel/p5-Locale-gettext
/usr/ports/lang/perl5.34
/usr/ports/devel/p5-Locale-libintl
/usr/ports/converters/libiconv
/usr/ports/converters/p5-Text-Unidecode
/usr/ports/textproc/p5-Unicode-EastAsianWidth
/usr/ports/devel/autoconf-switch
/usr/ports/devel/automake
/usr/ports/math/mpdecimal
/usr/ports/devel/readline
/usr/ports/devel/libtool