Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

cmake: Pass OPTIMIZATION_FLAG to linker too #87657

Open
wants to merge 2 commits into
base: main
Choose a base branch
from

Conversation

keith-packard
Copy link
Collaborator

The compiler may want to know the desired optimization level during linking, as when the compiler multilib configuration includes -Os as a selector.

@zephyrbot zephyrbot added size: XS A PR changing only a single line of code area: Build System labels Mar 25, 2025
@zephyrbot zephyrbot added the area: Toolchains Toolchains label Mar 25, 2025
@zephyrbot zephyrbot requested a review from stephanosio March 25, 2025 23:19
Copy link
Collaborator

@tejlmand tejlmand left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks for the cleanup.

It would be nice to get rid of -L<libgcc_dir>, but we should ensure there are no side-effects (can't remember in which cases this happend last time I looked at this).

A couple of code observations to be addressed in addition.

@@ -233,6 +233,7 @@ endif()
# Apply the final optimization flag(s)
zephyr_compile_options($<$<COMPILE_LANGUAGE:C>:${OPTIMIZATION_FLAG}>)
zephyr_compile_options($<$<COMPILE_LANGUAGE:CXX>:${OPTIMIZATION_FLAG}>)
add_link_options(${OPTIMIZATION_FLAG})
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This could be problematic as not all linkers may support the same optimization flags for compilation and linking.

Linkers invoked through the compiler, such as ld invoked through gcc/clang, will work but linkers invoked directly may fail, such as armlink.

@RobinKastberg how will the IAR linker behave in this regard ?

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yeah, I'm not sure how to ensure this doesn't break other toolchains. We already assume the linker can cope with a bunch of compiler flags, and most appear to want LINKERFLAGPREFIX. We could let the compiler tell the top level to not include ${OPTIMIZATION_FLAG} in the linker options? That would at least provide a simple way to fix problems if they crop up?

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We already assume the linker can cope with a bunch of compiler flags, and most appear to want LINKERFLAGPREFIX.

Do we ?
This is one reason why we have linker flags properties, so that each linker can translate a property name into a linker option for the linker in use, like this:

set_property(TARGET linker PROPERTY lto_arguments -flto=auto -fno-ipa-sra -ffunction-sections -fdata-sections)

add_link_options($<TARGET_PROPERTY:linker,lto_arguments>)

Or by setting the option in a specific compiler/linker combo cmake file, like gcc/ld:

add_link_options(-gdwarf-4)

There could still be legacy occurrences left which have not been updated, but I guess that just means those have proved themselves to be working until now, but that doesn't mean we should introduce new ones.

The main problem at this line, is that you take a flag defined for the compiler, and assume it works for the linker.
It might work for some, but can't be guaranteed to work for all.
Ref:

get_property(OPTIMIZE_FOR_SPEED_FLAG TARGET compiler PROPERTY optimization_speed)

set(OPTIMIZATION_FLAG ${OPTIMIZE_FOR_SPEED_FLAG})

add_link_options(${OPTIMIZATION_FLAG})

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm open to suggestions, but we have to find a way to get the complete set of flags necessary to specify the correct linker paths to the linker, and that includes a bunch of flags which are currently only passed to the compiler. We could create linker versions of all of these flags, but many of those are not specified using target properties today, so that would involve a fairly extensive change to existing projects.

Alternatively, we could create a linker function that took these compiler options and mapped them to appropriate linker options, filtering out those which weren't meaningful. A default function would discard all of them. Something like 'linker_add_compiler_options'?

@keith-packard keith-packard force-pushed the link-optimization-flag branch from 89450e4 to ca2f7c8 Compare March 26, 2025 20:01
@keith-packard
Copy link
Collaborator Author

I've reworked the second patch so that the computation of rt_library and lib_include_dir are delayed until the compiler flags are finished.

I renamed library_find_path to compiler_find_path and then placed the gcc/clang implementation of that along with a new function, compiler_set_linker_properties into cmake/compiler/target_template.cmake, which gets included just after cmake/linker/target_template.cmake from FindTargetTools.cmake.

With this, the values of lib_include_dir and rt_library are both uniformly computed using --print-libgcc-file-name. As lib_include_dir is already used in the linker, that will ensure that the -L flag continues to get passed along as before, but this time it will have a value computed using the whole set of compiler flags.

This refactoring let me remove a bunch of nearly identical code in the compiler target.cmake files.

@keith-packard keith-packard force-pushed the link-optimization-flag branch from ca2f7c8 to 6649017 Compare March 27, 2025 16:47
The compiler may want to know the desired optimization level during
linking, as when the compiler multilib configuration includes -Os as a
selector.

Signed-off-by: Keith Packard <keithp@keithp.com>
@keith-packard keith-packard force-pushed the link-optimization-flag branch from 6649017 to 09a5f46 Compare March 28, 2025 00:07
@keith-packard
Copy link
Collaborator Author

It would be nice to get rid of -L<libgcc_dir>, but we should ensure there are no side-effects (can't remember in which cases this happend last time I looked at this).

Ok, I've discovered it's actually impossible for us to compute libgcc_dir correctly in this fashion because some of the critical compiler flags are only available in the INTERFACE_COMPILE_OPTIONS property which is a generator expression, and hence cannot be evaluated during the cmake run.

Things like -m32/-m64, various -mfpu flags and many flags used on arc targets will all affect the multilib path value, but are unavailable when computing the libgcc path. I'm kinda amazed this ever worked anywhere.

For libgcc/compiler-rt, it's pretty simple - we can figure out what the runtime library basename is and use just that, without attempting to include the directory name. That should work as we can probably assume that any compiler install will use the same library name for all multilib configs?

For crtbegin.o/crtend.o, the linker won't search along the paths for them, so we really need to compute the whole path name. And that means capturing the output of the compiler front-end with the full compiler flags evaluated at build time?

With inclusion of the optimization flag into the multilib selection
process, we cannot compute the compiler library path when the compiler's
target.cmake is processed as OPTIMIZATION_FLAG is not computed until much
later.

Instead, add a function (compiler_file_path) which can be used to locate
the appropriate crtbegin.o and crtend.o files.

Delay computation of lib_include_dir and rt_library until after all
compiler flags have been computed by adding compiler_set_linker_properties
and calling that just before toolchain_linker_finalize is invoked.

Place default implementations of both of these functions in a new file,
cmake/compiler/target_template.cmake, where we assume the compiler works
like gcc or clang and handlers the --print-file-name and
--print-libgcc-file-name options. Compilers needing alternate
implementations can override these functions in their target.cmake files.

These implementations require that no generator expressions are necessary
for the compiler to compute the right library paths.

Signed-off-by: Keith Packard <keithp@keithp.com>
@keith-packard keith-packard force-pushed the link-optimization-flag branch from 09a5f46 to 3305ccf Compare March 28, 2025 01:15
@keith-packard
Copy link
Collaborator Author

Ok, I've discovered it's actually impossible for us to compute libgcc_dir correctly in this fashion because some of the critical compiler flags are only available in the INTERFACE_COMPILE_OPTIONS property which is a generator expression, and hence cannot be evaluated during the cmake run.

I did the best I could -- elided any flags which contained generator expressions and used the rest for all of the computations. That appears to at least get the flags needed for native_sim to work. A more complete solution would involve pushing the flag computation into the actual build run, and that seems 'problematic' as we'd have to pull the result back into the build process and insert it into the compile commands. Possible with make, but I'm unsure that could even work with ninja.

@57300 57300 removed their request for review April 1, 2025 19:05
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
area: Build System area: Toolchains Toolchains size: XS A PR changing only a single line of code
Projects
None yet
Development

Successfully merging this pull request may close these issues.

3 participants