1

TL;博士

我正在尝试构建一个使用此依赖项的 go 项目:https://github.com/mqu/openldap,它又从外部链接 lldap 和 llber 库,后者又使用 lgnutls,它使用 lnettle,这是我卡住了。

go build生成一长串未定义的引用并且构建失败。这是一个示例:

/usr/lib/x86_64-linux-gnu/libgnutls.a(sha-x86-ssse3.o): In function `_ctx_init':
(.text+0x468): undefined reference to `nettle_sha256_digest'
/usr/lib/x86_64-linux-gnu/libgnutls.a(sha-x86-ssse3.o): In function `_ctx_init':
(.text+0x476): undefined reference to `nettle_sha224_init'
/usr/lib/x86_64-linux-gnu/libgnutls.a(sha-x86-ssse3.o): In function `_ctx_init':
(.text+0x494): undefined reference to `nettle_sha256_init'
/usr/lib/x86_64-linux-gnu/libgnutls.a(sha-x86-ssse3.o): In function `_ctx_init':
(.text+0x4b8): undefined reference to `nettle_sha256_digest'
/usr/lib/x86_64-linux-gnu/libgnutls.a(sha-x86-ssse3.o): In function `_ctx_init':
(.text+0x4c6): undefined reference to `nettle_sha256_init'
/usr/lib/x86_64-linux-gnu/libgnutls.a(sha-x86-ssse3.o):(.data.rel.ro+0x18): undefined reference to `nettle_sha256_init'
/usr/lib/x86_64-linux-gnu/libgnutls.a(sha-x86-ssse3.o):(.data.rel.ro+0x28): undefined reference to `nettle_sha256_digest'
/usr/lib/x86_64-linux-gnu/libgnutls.a(sha-x86-ssse3.o):(.data.rel.ro+0x58): undefined reference to `nettle_sha224_init'
/usr/lib/x86_64-linux-gnu/libgnutls.a(sha-x86-ssse3.o):(.data.rel.ro+0x68): undefined reference to `nettle_sha224_digest'
/usr/lib/x86_64-linux-gnu/libgnutls.a(sha-x86-ssse3.o):(.data.rel.ro+0x98): undefined reference to `nettle_sha1_init'
/usr/lib/x86_64-linux-gnu/libgnutls.a(sha-x86-ssse3.o):(.data.rel.ro+0xa8): undefined reference to `nettle_sha1_digest'

我的go build命令:

CC=/usr/local/musl/bin/musl-gcc \
    GOOS=linux go build \
    -o /bin/activedirectory \
    -ldflags '-linkmode external -extldflags "-static -L/usr/lib/x86_64-linux-gnu -lnettle -lp11 -lsasl2 -lgnutls -ltasn1"'

我尝试通过安装libnettle4, nettle-dev, libghc-nettle-dev,来解决这个问题nettle-bin。我已确保包含-lnettle在 ldflags 中。没运气。

更多上下文

openldap 库正在链接代码中的ldaplber库:

package openldap

/*

#define LDAP_DEPRECATED 1
#include <stdlib.h>
#include <ldap.h>

static inline char* to_charptr(const void* s) { return (char*)s; }
static inline LDAPControl** to_ldapctrlptr(const void* s) {
    return (LDAPControl**) s;
}
*/
// #cgo CFLAGS: -DLDAP_DEPRECATED=1
// #cgo linux CFLAGS: -DLINUX=1
// #cgo LDFLAGS: -lldap -llber

这要求我安装并在 ldflags 中包含您在我的构建命令中看到的所有库。

简而言之,依赖链是这样的:

lldap -> lgnutls -> lnettle。

我添加了 lgnutls 并解决了我的 gnutls 依赖问题,但我无法解决我的荨麻依赖问题。

我的问题

我在尝试解决这些荨麻依赖性问题时做错了什么?

奖金问题

是否有解决这些 ld 链接器依赖项的最佳实践?现在我的流程是这样的:

  1. 运行构建
  2. 注意“未定义的引用”错误
  3. 通过谷歌搜索错误或查看引用的名称(例如nettle_sha1_digest=荨麻包)找出缺少哪个包
  4. 安装那个包
  5. 重复

我想我想知道是否有一个神奇的子弹可以为我安装所有依赖项?:)

4

1 回答 1

1

从外观上看,您以错误的顺序链接依赖库(有关更多上下文,请参阅为什么链接库的顺序有时会导致 GCC 中的错误?)。

对于静态链接库,链接器尝试按照指定库的顺序解析符号。如果依赖符号出现为未定义,则链接器查看它是否在指定的后续库中定义。由于在您的构建调用-lnettle中指定gnutls 库无法从荨麻解析它需要的符号之前 -lgnutls

这意味着(至少)您必须将引用移至-lnettleafter -lgnutls。不确定这会解决您的所有问题,因为我不熟悉您列出的所有链接依赖项,但它至少应该解决您当前运行的错误go build

于 2017-11-16T06:51:32.323 回答