Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

CMake link a shared library to static libraries

Tags:

c++

c

cmake

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)

like image 322
Velkan Avatar asked Jul 19 '15 16:07

Velkan


People also ask

Can shared library be statically linked?

You can't statically link a shared library (or dynamically link a static one).

What is static and shared library?

Static libraries take longer to execute, because loading into the memory happens every time while executing. While Shared libraries are faster because shared library code is already in the memory.


2 Answers

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.

like image 106
Tsyvarev Avatar answered Sep 28 '22 06:09

Tsyvarev


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
like image 45
zaufi Avatar answered Sep 28 '22 06:09

zaufi