gpt4 book ai didi

autotools - 如何将自动工具配置为仅在安装依赖项时才构建可选组件?

转载 作者:行者123 更新时间:2023-12-03 08:47:00 26 4
gpt4 key购买 nike

我有一个使用 autotools 作为构建系统的 C++ 项目。

我的程序涉及一个抽象接口(interface),其中有多个实现,每个实现都在单独的源文件中。这些实现取决于可能安装也可能不安装的库,具体取决于用户偏好和平台。

我想配置自动工具来跳过构建缺乏底层依赖项的实现。我愿意以用户和包维护者通常期望的任何方式来这样做。

举个例子,假设我的程序输出音频,它可以通过 OSS 来实现。 , ALSA ,或JACK 。这些分别在 oss_output.ccalsa_output.ccjack_output.cc 中实现。

目前我的 Makefile.am 包含(以及其他一些内容):

myprog_SOURCES = \
src/oss_output.cc \
src/alsa_output.cc \
src/jack_output.cc

myprog_LDFLAGS = [...some other stuff...] -ljack

不幸的是,尝试在未安装 JACK 的系统上构建此程序会失败,因为链接器找不到 libjack 和/或编译器因缺少 JACK 的头文件而失败。如果未安装 JACK,我会从 myprog_SOURCES 中删除 src/jack_output.cc,并从 myprog_SOURCES 中删除 -ljack myprog_LDFLAGS。 ALSA 和 OSS 也类似。

在这些依赖项(OSS、ALSA 和 JACK)中,其中一些库使用 pkg-config。有些没有,但我仍然可以通过头文件和库的存在来检测它们的存在。

如何使用自动工具实现这一点,并注意遵循最佳实践和惯例?

最佳答案

How is this accomplished with autotools, taking care to follow best practice and convention?

有两部分:

  1. 检测需要/支持哪些子系统,以及
  2. 将构建限制为仅选定的子系统。

确定要支持的子系统

第一 block 是 Autoconf 的面包和黄油。通常,人们会 check for the presence of each subsytem's library(-ies)通过 AC_CHECK_LIB()AC_SEARCH_LIBS() 来识别它是否可用,以及 check for the main header file(s)通过 AC_CHECK_HEADER()AC_CHECK_HEADERS()

建立一种机制来选择退出对某些可用子系统的构建支持将会很方便。这将采取设置 --without 的形式或--disable论据(选择哪一个通常是不明确的)。这不是强制性的,但我认为这是一种很好的形式。但请注意,确实需要小心谨慎,以避免在链接中包含所有找到的库。

凯迪拉克版本还将为用户提供支持,指定在哪里找到所需的库和 header ,以防它们不在默认搜索路径中。我经常在维护良好的项目中看到这种情况,但它使复杂性上升了一个层次,而且我不认为它以任何方式是强制性的。

实现细节可能会有很大差异,但通常情况下,对于每个子系统来说,最终结果是:

  • 一个 shell 变量,指示是否在构建中包含子系统(例如,可能是 build_oss),以及
  • (可选)输出变量(通过 AC_SUBST() 创建)传送预处理器/编译器/和/或链接器标志,用于针对适当的后端库进行构建。例如,OSS 部分的部分或全部 OSS_CPPFLAGSOSS_CXXFLAGSOSS_LIBS

将构建限制为选定的子系统

Automake 和 Autoconf 在这方面协同工作。关键工具是Automake conditional 。 Automake 提供了一个 Autoconf 宏 AM_CONDITIONAL,用于定义这些宏并为其赋值,并且它支持 if/then/else 基于它们在 Makefile.am 文件中构建。 (请注意,尽管它们具有相似的语法,但它们完全独立于 GNU make 条件。)

我链接的 Automake 手册的部分引用了其他几个讨论如何在各种场景中使用条件的部分,但并不完全清楚其中哪些适用于您。然而,我最好的猜测是,您真正需要的是 conditional sources ,所以这就是我要关注的重点。

Autoconf 部分将继承我已经讨论过的部分。它可能很简单

AM_CONDITIONAL([oss], [test "$build_oss" = "1"])
AM_CONDITIONAL([alsa], [test "$build_alsa" = "1"])
AM_CONDITIONAL([jack], [test "$build_jack" = "1"])

Automake 手册给出了其他示例。

Makefile.am 方面,可能与如下内容匹配:

myprog_SOURCES = \
src/main.cc \
src/another.cc \
...

if oss
myprog_SOURCES += src/oss_output.cc
endif
if alsa
myprog_SOURCES += src/alsa_output.cc
endif
if jack
myprog_SOURCES += src/jack_output.cc
endif

您还需要注意应用每个子系统的标志和库。您可以采用与源相同的方法(甚至使用相同的 if block ),那就好了,或者您可以让 configure 通过以下方式进行控制(是否)首先为每个子系统定义任何标志。

您会注意到我掩盖了许多细节。这是对我必须视为高级别问题的相当高级别的答案。如果我以其他方式回答这个问题,那么这个问题对于这个 field 来说就太宽泛了。

关于autotools - 如何将自动工具配置为仅在安装依赖项时才构建可选组件?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/61030804/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com