Post

Building FRR Notes

最近在一台 Ubuntu 上编译 frr, 发现编译之后运行总是提示找不到文件路径 /etc/frr.conf,而该路径确实没有这个文件,后来发现 frr.conf 安装后的实际路径是 /etc/frr。 看代码发现当前编译的版本是9.1.3,而之前按照同样的方法编译和运行成功的版本是10.2。难道和版本有关,经过一番分析和查找,终于发现 frr 官网上Ubuntu22.04 的构建方法是针对最新的版本(写作本文时是10.3)。 其中提到在执行 ./configure 时,配置的是:

1
2
    --sysconfdir=/etc \
    --localstatedir=/var \

而在9.2版本之前,当然也包括9.1.3,在执行 ./configure 时,配置应该的是

1
2
    --sysconfdir=/etc/frr \
    --localstatedir=/var/run/frr

否则就会出现运行 frr 时,本文开始提到的错误信息。对比上面配置的差异,可以看出 sysconfdir 和 localstatedir 的路径设置发生了变化。原因在最新代码的configure.ac中已经给出了说明:

system paths

Versions of FRR (or Quagga, or Zebra) before ca. 9.2 used sysconfdir and localstatedir as-is, without appending /frr. The /frr was expected to be given on ./configure invocations.

This does not match standard behavior by other packages and makes FRR specific packaging changes necessary to add these options. localstatedir was also misused to include the /run part (it normally is only /var), leaving no path configuration option that references /var itself. This is because runstatedir did not exist in ancient autoconf.

The path options have been changed to expect plain / system prefix directories. As a temporary workaround to not break packaging, eventual /frr suffixes are stripped and a warning is printed.

简单来说就是大概在9.2版本之前的 sysconfdir 和 localstatedir,需要手动在执行 ./configure 时候配置 frr 完整路径,而这种方式和其他包使用的标准方式不同。同时因为 localstatedir 错误的包含了 /run 这一部分,导致没有路径可以指向 /var 自己。

在上面 10.0 版本修改配置路径后,为了兼容性的考虑,如果用户按照之前的方式配置 configure, 则脚本中会自动去除末尾的 /frr,保证 sysconfdir 和 localstatedir 只包含指定目录,这通常是指路径 /etc 和 /var,同时打印一条告警提示信息。

查看 configure.ac 文件的修改历史,发现这个 merge 是在 2024年1月28日合入的。commit信息如下:

build: untangle sysconfdir & localstatedir --sysconfdir should be /etc and --localstatedir should be /var. The package-specific subdirectory should be added by configure, not given by the user, to match established behavior by other packages.

Note that --bindir, --sbindir, --libdir and --libexecdir have different established/expected behavior due to distro specific multi-arch support. That’s why these are left unchanged.

The reason this is getting fixed now is that we need to use --localstatedir for its actual value to put things in /var/lib. As it is now, being overloaded for /run, the configured /var path becomes inaccessible.

上面也提到了因为要用 localstatedir 来引用 /var/lib,这在新版本可以直接实现,而 9.2 之前的版本是没法直接访问的。

综上所述,使用 9.2 之前的 configure 配置可以在所有版本上正常编译运行 frr,因为 10.0 之后的编译脚本考虑了兼容性。但是使用最新的 configure 编译指导文档去配置 9.2 之前的 frr 代码,则在安装后运行时,会出现上面提到的 frr 运行时找不到 /etc/frr.conf文件的问题。这其实也是一个兼容性问题,因为很可能有人用最新的文档去编译 9.2 之前的版本,那样也会遇到本文开始提到的问题。 既然旧代码已经改不了了,那么最新的文档最好将这点说明一下。

This post is licensed under CC BY 4.0 by the author.