CMake link a shared library to static libraries

16,330

Solution 1

As I know, CMake doesn't allow to mix STATIC and SHARED libraries.

If your staticfoo library is used solely as part of other libraries/executables, you can define it as

add_library(staticfoo OBJECT <src>)

and use then as some sort of sources when build other library:

add_library(sharedfoo SHARED <src> $<TARGET_OBJECTS:staticfoo>)

For more info see documentation on add_library.

Solution 2

To resolve this case you need to do few things:

  • first of all make sure you've compiled static libraries w/ -fPIC, so they'll contain a relocatable code (which would be a part of a shared library later)
  • then, you need to control symbols visibility when compiling all libraries, so being a part of shared library, symbols came from the static one would be visible
  • and finally, yes, you need to specify PRIVATE <static libs> when linking your shared library, so the linker command line for your executable wouldn't have any static libs

Solution 3

See my answer here. Basically add /WHOLEARCHIVE, -all_load, or --whole-archive to the linker flags.

Share:
16,330
Velkan
Author by

Velkan

Updated on July 19, 2022

Comments

  • Velkan
    Velkan almost 2 years

    I am porting an AutoTools project to CMake.

    What AutoTools does:

    • builds some static libraries
    • builds some shared libraries and links static ones into shared
    • builds an executable, links it to shared libraries

    What I've managed to do with CMake:

    • build some static libraries - add_library(staticfoo <src>)
    • build some shared libraries - add_library(sharedfoo SHARED <src>) and link them - target_link_libraries(sharedfoo staticfoo)
    • build an executable, link it to shared libraries - target_link_libraries(exe sharedfoo), but that dragged the static libraries in again, too.

    So, the resulting link command for the executable has static libs in addition to shared. Which doesn't correspond to the command generated by AutoTools project.

    I've tried target_link_libraries(sharedfoo PRIVATE staticfoo), but that doesn't get the symbols from the static lib into the interface of the shared lib.

    How to get the symbols without that 'transitive' behavior?

    (in platform-independent way)

  • zaufi
    zaufi almost 9 years
    CMake don't care what kind of libraries you've tried to link static/dynamic/mixed... that would be a problem of a linker at particular platform.
  • Youka
    Youka almost 9 years
    On windows, the linker discards export symbols by linking archieved objects/static libraries, but unpacked objects work. That's one of the main reasons why object libraries were added to cmake.
  • Velkan
    Velkan almost 9 years
    So, how to 'control symbols visibility'?
  • tamas.kenez
    tamas.kenez almost 9 years
    Indeed this is a good way to convert that pattern found in autotools projects. For alternatives and their issues see this well-detailed question: stackoverflow.com/questions/5136184/…
  • zaufi
    zaufi almost 9 years
    most obvious way is to use -fvisibility compiler option. CMake >= 3.x has CMAKE_<LANG>_VISIBILITY_PRESET variable (and corresponding target property). But unfortunately it doesn't affect static libraries. That is a lack and will be fixed in upcoming release of CMake 3.3.
  • Chris_128
    Chris_128 almost 3 years
    Additionally, I needed to enable position independent code for the static library using set_property(TARGET staticfoo PROPERTY POSITION_INDEPENDENT_CODE ON). See stackoverflow.com/questions/38296756/….