So I'm working on a project where I have the following, simplified structure
./
┗━ utils
┗━ BUILD
┗━ tool1
┗━ BUILD
┗━ itk.BUILD
┗━ WORKSPACE
in essence i'm trying to build a binary tool1
that depends on a library utils
which in turn depends on itk
here are the contents of the utils/BUILD
file (simplified):
cc_library(
name = "utils",
srcs = glob(
[
"utils/*.cpp",
"include/**",
],
),
hdrs = glob([
"include/*.h",
]),
copts = [
"-DITK_USE_SYSTEM_ZLIB",
],
includes = ["./include"]
linkopts = ["-ldl", "-lz", "-lm", "-lgomp", "-lpthread", "-lcrypto", "-lrt"],
visibility = ["//visibility:public"],
deps = [
"@itk//:itk_libs",
],
)
and tool1/BUILD
:
cc_binary(
name = "tool1",
srcs = ["tool1.cpp"],
deps = [
"//:utils",
],
)
ITK has been pre-compiled in say third_party/itk
, it's included as a local repo in the WORKSPACE
file:
new_local_repository(
name = "itk",
path = "third_party/itk",
build_file = "itk.BUILD",
)
itk.BUILD
just creates a collective library:
cc_library(
name = "itk_libs",
srcs = glob(["lib/**/*.a"]),
hdrs = glob([
"include/ITK-4.13/**/*.h",
"include/ITK-4.13/*.h",
"include/ITK-4.13/**/*.hxx",
"include/ITK-4.13/*.hxx",
"include/ITK-4.13/**/*.txx",
"include/ITK-4.13/*.txx",
]),
strip_include_prefix = "include/ITK-4.13",
visibility = ["//visibility:public"],
)
when building utils using bazel bazel build //:utils
it succeeds, however, when running bazel build //tool1
i get a bunch of link errors like this one:
external/itk/lib/libitkvnl_algo-4.13.a(vnl_cholesky.cxx.o): In function `vnl_cholesky::vnl_cholesky(vnl_matrix<double> const&, vnl_cholesky::Operation)':
vnl_cholesky.cxx:(.text+0x99): undefined reference to `v3p_netlib_dpofa_'
I'm a the end of my wits, been struggling with this for a few weeks now.
What I don't understand is that //:utils
compiles successfully, the link errors only occur when building //tool1
.
I tried to change the order of the libraries in the command line and run it manually, same results.
I tried to add itk directly as a dependency to tool1, that is add @itk//:itk_libs
to deps
in tool1/BUILD
, same results
I must be missing something, I would appreciate it if someone could point me to the right direction. I'm almost certain the problem is something with ITK that I'm unaware of, not necessarily with bazel.
Edit: I was getting a bunch of other errors from other dependecies, which are now resolved, curiously the only errors remaining are all related to libitkvnl_algo-4.13.a.
Edit 2 (for reference after accepting an answer):
I wasn't clear on some bits, the missing references were all from ITK's own libraries, which were all compiled and existing in the command line that's produced by bazel. Also I tried reordering one library in the command line before asking the question, while that didn't work at first, now after Paul's answer I realised, it wasn't just one library that needs reordering.
If the archive libitkvnl_algo.a
depends on symbols defined in libv3p_netlib.a
, then (on Linux) the order of the libraries on the link line for creating the binary matters: libv3p_netlib.a
must appear after libitkvnl_algo.a
. You won't necessarily get a usable ordering if you just use glob
; you may have to explicitly list the libraries in the srcs
. Or you can construct one cc_library
per .a
file and use deps
to reflect the correct dependency graph; then bazel will give you the correct link-line ordering automatically.