[CMake] Link order and interface multiplicity

Etan Kissling kissling at oberon.ch
Mon May 15 08:30:10 EDT 2017


Heya,

I have a project with a layer consisting of interface libraries:
	add_library(I INTERFACE)

These interface libraries are then implemented several times, to fit the different environments of applications:
	add_library(IA STATIC ...)
	target_link_libraries(IA PUBLIC I)

	add_library(IB STATIC ...)
	target_link_libraries(IB PUBLIC I)

There are also application independent libraries, that make use of the interface libraries.
	add_library(Foo STATIC ...)
	target_link_libraries(Foo PUBLIC I)

	add_library(Bar STATIC ...)
	target_link_libraries(Bar PUBLIC I)

And finally, the application defines which implementation of the interface library layer is being used.
	add_executable(ExeA ...)
	target_link_libraries(ExeA Foo Bar IA)

	add_executable(ExeB ...)
	target_link_libraries(ExeB Foo Bar IB)


Luckily, this is okay, as long as IA is listed after Foo and Bar in the synthesized link command.

However, certain implementations of I make use of the application independent libraries again.
On these environments, the link command line becomes something like this:
	IA Foo Bar

While it should be
	Foo Bar IA Foo Bar

This make sense, because there is no explicit dependency being described that Foo / Bar depend on IA while compiling ExeA.
In the simple case, we just get lucky, because it happens to be the default that link command line has the same order as in the target_link_libraries call.

I'm working with gcc-arm-none-eabi cross compiler. Here's what I've tried:
• LINK_INTERFACE_MULTIPLICITY 
	Does not seem to work. The generated command line is still the same even with higher numbers.
• set(CMAKE_EXE_LINKER_FLAGS "${CMAKE_EXE_LINKER_FLAGS} -Wl,--start-group")
	Linking succeeds, but the program does not run properly on the target hardware. Only after connecting a debugger and resetting, it runs. Strange behaviour.
	When I link Foo / Bar explicitly with IA instead of --start-group, the command line becomes Foo Bar IA Foo Bar, and then everything works fine. But I cannot do this in general, because of ExeB ^^


=> Is there a way how I can define that Foo / Bar temporarily depend on IA while compiling ExeA, and temporarily depend on IB while compiling ExeB?
=> Is there a different approach on how to handle project structured like the one described above?

Thanks

Etan


More information about the CMake mailing list