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)
See my answer here. Basically add
/WHOLEARCHIVE
,-all_load
, or--whole-archive
to the linker flags.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 asand use then as some sort of sources when build other library:
For more info see documentation on add_library.
I created a little example that does this the right way, here: https://github.com/CarloWood/cmaketest
It shows how two "static" libraries, one with two visible symbols, both with by default hidden symbols, are added to a shared library, which is then used to link an executable.
The method used is that of @Tsyvarev, combined with the VISIBILITY target property mentioned by @Zaufi.
The only non-platform specific thing (I think) is the __EXPORT macro that is used. It would be nice to fix that too, if anyone knows how.
To resolve this case you need to do few things:
-fPIC
, so they'll contain a relocatable code (which would be a part of a shared library later)PRIVATE <static libs>
when linking your shared library, so the linker command line for your executable wouldn't have any static libs