I'm using CMake in a project, and I'm trying to statically link some libraries. I've set:
set(BUILD_SHARED_LIBS OFF)
set(CMAKE_EXE_LINKER_FLAGS "-static-libgcc -static-libstdc++ -static")
set_target_properties(icarus PROPERTIES LINK_SEARCH_END_STATIC 1)
And I've made sure when looking for the actual libraries that I have the *.a version of them.
Currently the project imports:
libPocoNet.a
libPocoUtil.a
libPocoXML.a
libPocoFoundation.a
libmysqlclient.a
libmysqlpp.a
libcrypto++.a
CUDA
All libraries are found, and when doing a dynamic/shared linkage, they work fine. I have also tried to set compilation flags:
set(GCC_CXX_FLAGS ${GCC_CXX_FLAGS} "-static-libgcc -static-libstdc++ -static")
But to no avail. While I get no issues while compiling, linkage is throwing alot of undefined reference errors for calls found in the above libraries, i.e:
undefined reference to `mysql_thread_init'
undefined reference to `mysql_real_query'
undefined reference to `pthread_mutex_unlock'
undefined reference to `Poco::ErrorHandler::handle()'
Not in that particular order, and numerous errors for each library.
Looking at the last line of GCC I see:
/usr/bin/c++ -g -g -static-libgcc -static-libstdc++ -static [list of *.cpp files]
-o icarus -rdynamic /usr/local/lib/libPocoFoundation.a /usr/local/lib/libPocoNet.a
/usr/local/lib/libPocoUtil.a /usr/local/lib/libPocoXML.a
-Wl,-Bstatic -lmysqlclient -lmysqlpp -lcrypto++
Which makes me wonder:
- Why are Poco libraries linked as -rdynamic, and there is no -Wl -Bstatic flag? As if they are skipped/excluded from static linkage.
- mysqlclient, mysqlpp and crypto++ seem to be set for static linkage, yet I still get errors
So, could someone please explain to me:
- How do I setup for partial static linkage using CMake
- Is CMAKE_EXE_LINKER_FLAGS the only one I need to set?
- Should I be forcing static linking for mentioned libraries but not for entire project?
Please excuse me if those are too many or too localized questions, I haven't tried this before, and I can't seem to find much info on the net.
I've managed to solve my problem by using the following:
This works without passing a
-static
which creates other big issues, and can essentially mix static and dynamic libraries.As long as the order of static libraries is correct, and as long as dependencies of static libraries are satisfied, I get an ELF which loads what are are dynamic (i.e. in my case mysqlclient, libmysql++) and static all the rest (crypto++, PocoNet, PocoUtil, PocoXML, PocoFoundation).
Bear in mind that static linked libraries have their own dependencies. Examining my debug application using readelf -d app, I see:
I know that pthread is imported by Poco::Runnable, libm is for math operations, etc. I am still unaware if this is the right way to use CMake for partial static linking.
In the case of Debian packaged libraries, such as crypto++, mysql++, mysqlclient, simply finding the *.a library worked, but in case of Poco Libraries, which only got me the full path and name of the library, but not a flag,
-Bdynamic
could only be turned off by using the above lines.Note: Poco could not be linked statically, without
-static-libstdc++
I hope this helps anyone stuck at something similar.
How do I setup for static linkage using CMake
Well... you don't :) That's not how CMake works: in CMake, you first find the absolute path of a library, then link to it with target_link_libraries.
So, if you want to link to a static library, you need to search for that static library:
instead of: