What's the difference between add_dependencies and target_link_libraries for library targets?

208 Views Asked by At

In CMake, what is the difference between:

add_dependencies(tgt1 tgt2)

and

target_link_libraries(tgt1 tgt2)

when tgt2 being a library target?

Note: I don't mean difference in implementation, I mean difference in semantics.

2

There are 2 best solutions below

1
On BEST ANSWER

add_dependencies just makes the second target's build get brought up to date before the first target if both targets need to be brought up to date. It's a build order control mechanism.

target_link_libraries makes the second target a link dependency of the first target- they will be linked by the generated buildsystem- and makes it so that if the depended-upon target needs to be rebuilt, the first target will also be rebuilt if need by. When using PRIVATE or PUBLIC visibility, it's to express and instruct that things that are in the interface of the second target will be made available to the first target. When using INTERFACE or PUBLIC visibility, it's to express and instruct that things that are in the interface of the second target will be made available to targets that express link dependencies on the first target ("transitive" link dependencies).

0
On

add_dependencies(A B) makes sure the generated project will make sure B is up to date before building A. It's the equivalent to adding the following rule to a makefile

A: B

target_link_libraries(A B) has the same effect plus:

  • Appropriate linker flags are added to include the binary generated for B among the files linked when generating the binary for A
  • Several INTERFACE properties of B take effect on A. Many of those correspond to cmake commands you can pass INTERFACE/PUBLIC to, e.g.
    • target_compile_definitions
    • target_compile_features
    • target_compile_options
    • target_include_directories
    • target_link_directories
    • target_link_libraries
    • target_link_options
    • target_precompile_headers
    • target_sources
  • If A is a library and you're adding PUBLIC/INTERFACE as visibility, targets linking A get access to INTERFACE properties in a similar manner
  • Some additional linker options may be modified specified e.g. for setting the rpath
  • The behaviour of install(EXPORT) and install(TARGETS) may change slightly. (Export dependency information, rpath update.)
  • The LINK_LIBRARIES target property for A is changed
  • ...