2021-03-20 12:12:33 +08:00
|
|
|
function(collect_object_file_deps target result)
|
|
|
|
set(all_deps "")
|
|
|
|
get_target_property(target_type ${target} "TARGET_TYPE")
|
|
|
|
if(NOT target_type)
|
|
|
|
return()
|
2020-05-01 01:46:49 +08:00
|
|
|
endif()
|
|
|
|
|
2021-03-20 12:12:33 +08:00
|
|
|
if(${target_type} STREQUAL ${OBJECT_LIBRARY_TARGET_TYPE})
|
|
|
|
list(APPEND all_deps ${target})
|
|
|
|
get_target_property(deps ${target} "DEPS")
|
|
|
|
foreach(dep IN LISTS deps)
|
|
|
|
collect_object_file_deps(${dep} dep_targets)
|
|
|
|
list(APPEND all_deps ${dep_targets})
|
|
|
|
endforeach(dep)
|
|
|
|
set(${result} ${all_deps} PARENT_SCOPE)
|
2020-05-01 01:46:49 +08:00
|
|
|
return()
|
|
|
|
endif()
|
|
|
|
|
2021-03-20 12:12:33 +08:00
|
|
|
if(${target_type} STREQUAL ${ENTRYPOINT_OBJ_TARGET_TYPE})
|
|
|
|
set(entrypoint_target ${target})
|
|
|
|
get_target_property(is_alias ${entrypoint_target} "IS_ALIAS")
|
|
|
|
if(is_alias)
|
|
|
|
get_target_property(aliasee ${entrypoint_target} "DEPS")
|
|
|
|
if(NOT aliasee)
|
|
|
|
message(FATAL_ERROR
|
|
|
|
"Entrypoint alias ${entrypoint_target} does not have an aliasee.")
|
|
|
|
endif()
|
|
|
|
set(entrypoint_target ${aliasee})
|
2020-05-01 01:46:49 +08:00
|
|
|
endif()
|
2021-03-20 12:12:33 +08:00
|
|
|
list(APPEND all_deps ${entrypoint_target})
|
|
|
|
get_target_property(deps ${target} "DEPS")
|
|
|
|
foreach(dep IN LISTS deps)
|
|
|
|
collect_object_file_deps(${dep} dep_targets)
|
|
|
|
list(APPEND all_deps ${dep_targets})
|
|
|
|
endforeach(dep)
|
|
|
|
set(${result} ${all_deps} PARENT_SCOPE)
|
2020-05-01 01:46:49 +08:00
|
|
|
return()
|
|
|
|
endif()
|
2021-10-20 07:06:52 +08:00
|
|
|
|
|
|
|
if(${target_type} STREQUAL ${ENTRYPOINT_EXT_TARGET_TYPE})
|
|
|
|
# It is not possible to recursively extract deps of external dependencies.
|
|
|
|
# So, we just accumulate the direct dep and return.
|
|
|
|
get_target_property(deps ${target} "DEPS")
|
|
|
|
set(${result} ${deps} PARENT_SCOPE)
|
|
|
|
return()
|
|
|
|
endif()
|
2021-03-20 12:12:33 +08:00
|
|
|
endfunction(collect_object_file_deps)
|
2020-05-01 01:46:49 +08:00
|
|
|
|
2020-04-18 09:42:40 +08:00
|
|
|
# A rule to build a library from a collection of entrypoint objects.
|
|
|
|
# Usage:
|
|
|
|
# add_entrypoint_library(
|
|
|
|
# DEPENDS <list of add_entrypoint_object targets>
|
|
|
|
# )
|
2020-04-22 01:10:58 +08:00
|
|
|
#
|
2021-10-20 07:06:52 +08:00
|
|
|
# NOTE: If one wants an entrypoint to be available in a library, then they will
|
2020-04-22 01:10:58 +08:00
|
|
|
# have to list the entrypoint target explicitly in the DEPENDS list. Implicit
|
|
|
|
# entrypoint dependencies will not be added to the library.
|
2020-04-18 09:42:40 +08:00
|
|
|
function(add_entrypoint_library target_name)
|
|
|
|
cmake_parse_arguments(
|
|
|
|
"ENTRYPOINT_LIBRARY"
|
|
|
|
"" # No optional arguments
|
|
|
|
"" # No single value arguments
|
2021-10-20 07:06:52 +08:00
|
|
|
"DEPENDS" # Multi-value arguments
|
2020-04-18 09:42:40 +08:00
|
|
|
${ARGN}
|
|
|
|
)
|
|
|
|
if(NOT ENTRYPOINT_LIBRARY_DEPENDS)
|
|
|
|
message(FATAL_ERROR "'add_entrypoint_library' target requires a DEPENDS list "
|
|
|
|
"of 'add_entrypoint_object' targets.")
|
|
|
|
endif()
|
|
|
|
|
2020-04-22 01:10:58 +08:00
|
|
|
get_fq_deps_list(fq_deps_list ${ENTRYPOINT_LIBRARY_DEPENDS})
|
2021-03-20 12:12:33 +08:00
|
|
|
set(all_deps "")
|
2020-04-22 01:10:58 +08:00
|
|
|
foreach(dep IN LISTS fq_deps_list)
|
2020-04-18 09:42:40 +08:00
|
|
|
get_target_property(dep_type ${dep} "TARGET_TYPE")
|
2021-10-20 07:06:52 +08:00
|
|
|
if(NOT ((${dep_type} STREQUAL ${ENTRYPOINT_OBJ_TARGET_TYPE}) OR (${dep_type} STREQUAL ${ENTRYPOINT_EXT_TARGET_TYPE})))
|
2020-04-18 09:42:40 +08:00
|
|
|
message(FATAL_ERROR "Dependency '${dep}' of 'add_entrypoint_collection' is "
|
2021-10-20 07:06:52 +08:00
|
|
|
"not an 'add_entrypoint_object' or 'add_entrypoint_external' target.")
|
2020-04-18 09:42:40 +08:00
|
|
|
endif()
|
2021-03-20 12:12:33 +08:00
|
|
|
collect_object_file_deps(${dep} recursive_deps)
|
|
|
|
list(APPEND all_deps ${recursive_deps})
|
2020-04-18 09:42:40 +08:00
|
|
|
endforeach(dep)
|
2021-03-20 12:12:33 +08:00
|
|
|
list(REMOVE_DUPLICATES all_deps)
|
|
|
|
set(objects "")
|
|
|
|
foreach(dep IN LISTS all_deps)
|
2022-02-15 07:15:22 +08:00
|
|
|
list(APPEND objects $<$<STREQUAL:$<TARGET_NAME_IF_EXISTS:${dep}>,${dep}>:$<TARGET_OBJECTS:${dep}>>)
|
2021-03-20 12:12:33 +08:00
|
|
|
endforeach(dep)
|
2021-07-22 06:17:15 +08:00
|
|
|
|
2021-03-20 12:12:33 +08:00
|
|
|
add_library(
|
|
|
|
${target_name}
|
|
|
|
STATIC
|
|
|
|
${objects}
|
2020-04-18 09:42:40 +08:00
|
|
|
)
|
2021-05-10 15:53:48 +08:00
|
|
|
set_target_properties(${target_name} PROPERTIES ARCHIVE_OUTPUT_DIRECTORY ${CMAKE_CURRENT_BINARY_DIR})
|
2020-04-18 09:42:40 +08:00
|
|
|
endfunction(add_entrypoint_library)
|
|
|
|
|
|
|
|
# Rule to build a shared library of redirector objects.
|
|
|
|
function(add_redirector_library target_name)
|
|
|
|
cmake_parse_arguments(
|
|
|
|
"REDIRECTOR_LIBRARY"
|
|
|
|
""
|
|
|
|
""
|
|
|
|
"DEPENDS"
|
|
|
|
${ARGN}
|
|
|
|
)
|
|
|
|
|
|
|
|
set(obj_files "")
|
|
|
|
foreach(dep IN LISTS REDIRECTOR_LIBRARY_DEPENDS)
|
|
|
|
# TODO: Ensure that each dep is actually a add_redirector_object target.
|
|
|
|
list(APPEND obj_files $<TARGET_OBJECTS:${dep}>)
|
|
|
|
endforeach(dep)
|
|
|
|
|
|
|
|
# TODO: Call the linker explicitly instead of calling the compiler driver to
|
|
|
|
# prevent DT_NEEDED on C++ runtime.
|
|
|
|
add_library(
|
|
|
|
${target_name}
|
2021-02-24 23:22:58 +08:00
|
|
|
EXCLUDE_FROM_ALL
|
2020-04-18 09:42:40 +08:00
|
|
|
SHARED
|
|
|
|
${obj_files}
|
|
|
|
)
|
2021-05-10 15:53:48 +08:00
|
|
|
set_target_properties(${target_name} PROPERTIES LIBRARY_OUTPUT_DIRECTORY ${CMAKE_CURRENT_BINARY_DIR})
|
|
|
|
target_link_libraries(${target_name} -nostdlib -lc -lm)
|
|
|
|
set_target_properties(${target_name} PROPERTIES LINKER_LANGUAGE "C")
|
2020-04-18 09:42:40 +08:00
|
|
|
endfunction(add_redirector_library)
|
|
|
|
|
2020-04-22 01:10:58 +08:00
|
|
|
set(HDR_LIBRARY_TARGET_TYPE "HDR_LIBRARY")
|
|
|
|
|
[libc] Implement FLAGS option for generating all combinations for targets.
Add FLAGS option for add_header_library, add_object_library,
add_entrypoint_object, and add_libc_unittest.
In general, a flag is a string provided for supported functions under the
multi-valued option `FLAGS`. It should be one of the following forms:
FLAG_NAME
FLAG_NAME__NO
FLAG_NAME__ONLY
A target will inherit all the flags of its upstream dependency.
When we create a target `TARGET_NAME` with a flag using (add_header_library,
add_object_library, ...), its behavior will depend on the flag form as follow:
- FLAG_NAME: The following 2 targets will be generated:
`TARGET_NAME` that has `FLAG_NAME` in its `FLAGS` property.
`TARGET_NAME.__NO_FLAG_NAME` that depends on `DEP.__NO_FLAG_NAME` if
`TARGET_NAME` depends on `DEP` and `DEP` has `FLAG_NAME` in its `FLAGS`
property.
- FLAG_NAME__ONLY: Only generate 1 target `TARGET_NAME` that has `FLAG_NAME`
in its `FLAGS` property.
- FLAG_NAME__NO: Only generate 1 target `TARGET_NAME.__NO_FLAG_NAME` that
depends on `DEP.__NO_FLAG_NAME` if `DEP` is in its DEPENDS list and `DEP`
has `FLAG_NAME` in its `FLAGS` property.
To show all the targets generated, pass SHOW_INTERMEDIATE_OBJECTS=ON to cmake.
To show all the targets' dependency and flags, pass
`SHOW_INTERMEDIATE_OBJECTS=DEPS` to cmake.
To completely disable a flag FLAG_NAME expansion, set the variable
`SKIP_FLAG_EXPANSION_FLAG_NAME=TRUE`.
Reviewed By: michaelrj, sivachandra
Differential Revision: https://reviews.llvm.org/D125174
2022-05-09 01:45:40 +08:00
|
|
|
# Internal function, used by `add_header_library`.
|
|
|
|
function(create_header_library fq_target_name)
|
2020-04-18 09:42:40 +08:00
|
|
|
cmake_parse_arguments(
|
|
|
|
"ADD_HEADER"
|
[libc] Implement FLAGS option for generating all combinations for targets.
Add FLAGS option for add_header_library, add_object_library,
add_entrypoint_object, and add_libc_unittest.
In general, a flag is a string provided for supported functions under the
multi-valued option `FLAGS`. It should be one of the following forms:
FLAG_NAME
FLAG_NAME__NO
FLAG_NAME__ONLY
A target will inherit all the flags of its upstream dependency.
When we create a target `TARGET_NAME` with a flag using (add_header_library,
add_object_library, ...), its behavior will depend on the flag form as follow:
- FLAG_NAME: The following 2 targets will be generated:
`TARGET_NAME` that has `FLAG_NAME` in its `FLAGS` property.
`TARGET_NAME.__NO_FLAG_NAME` that depends on `DEP.__NO_FLAG_NAME` if
`TARGET_NAME` depends on `DEP` and `DEP` has `FLAG_NAME` in its `FLAGS`
property.
- FLAG_NAME__ONLY: Only generate 1 target `TARGET_NAME` that has `FLAG_NAME`
in its `FLAGS` property.
- FLAG_NAME__NO: Only generate 1 target `TARGET_NAME.__NO_FLAG_NAME` that
depends on `DEP.__NO_FLAG_NAME` if `DEP` is in its DEPENDS list and `DEP`
has `FLAG_NAME` in its `FLAGS` property.
To show all the targets generated, pass SHOW_INTERMEDIATE_OBJECTS=ON to cmake.
To show all the targets' dependency and flags, pass
`SHOW_INTERMEDIATE_OBJECTS=DEPS` to cmake.
To completely disable a flag FLAG_NAME expansion, set the variable
`SKIP_FLAG_EXPANSION_FLAG_NAME=TRUE`.
Reviewed By: michaelrj, sivachandra
Differential Revision: https://reviews.llvm.org/D125174
2022-05-09 01:45:40 +08:00
|
|
|
"" # Optional arguments
|
|
|
|
"" # Single value arguments
|
|
|
|
"HDRS;DEPENDS;FLAGS" # Multi-value arguments
|
2020-04-18 09:42:40 +08:00
|
|
|
${ARGN}
|
|
|
|
)
|
|
|
|
|
|
|
|
if(NOT ADD_HEADER_HDRS)
|
|
|
|
message(FATAL_ERROR "'add_header_library' target requires a HDRS list of .h files.")
|
|
|
|
endif()
|
|
|
|
|
|
|
|
set(FULL_HDR_PATHS "")
|
|
|
|
# TODO: Remove this foreach block when we can switch to the new
|
|
|
|
# version of the CMake policy CMP0076.
|
|
|
|
foreach(hdr IN LISTS ADD_HEADER_HDRS)
|
|
|
|
list(APPEND FULL_HDR_PATHS ${CMAKE_CURRENT_SOURCE_DIR}/${hdr})
|
|
|
|
endforeach()
|
|
|
|
|
[libc] Implement FLAGS option for generating all combinations for targets.
Add FLAGS option for add_header_library, add_object_library,
add_entrypoint_object, and add_libc_unittest.
In general, a flag is a string provided for supported functions under the
multi-valued option `FLAGS`. It should be one of the following forms:
FLAG_NAME
FLAG_NAME__NO
FLAG_NAME__ONLY
A target will inherit all the flags of its upstream dependency.
When we create a target `TARGET_NAME` with a flag using (add_header_library,
add_object_library, ...), its behavior will depend on the flag form as follow:
- FLAG_NAME: The following 2 targets will be generated:
`TARGET_NAME` that has `FLAG_NAME` in its `FLAGS` property.
`TARGET_NAME.__NO_FLAG_NAME` that depends on `DEP.__NO_FLAG_NAME` if
`TARGET_NAME` depends on `DEP` and `DEP` has `FLAG_NAME` in its `FLAGS`
property.
- FLAG_NAME__ONLY: Only generate 1 target `TARGET_NAME` that has `FLAG_NAME`
in its `FLAGS` property.
- FLAG_NAME__NO: Only generate 1 target `TARGET_NAME.__NO_FLAG_NAME` that
depends on `DEP.__NO_FLAG_NAME` if `DEP` is in its DEPENDS list and `DEP`
has `FLAG_NAME` in its `FLAGS` property.
To show all the targets generated, pass SHOW_INTERMEDIATE_OBJECTS=ON to cmake.
To show all the targets' dependency and flags, pass
`SHOW_INTERMEDIATE_OBJECTS=DEPS` to cmake.
To completely disable a flag FLAG_NAME expansion, set the variable
`SKIP_FLAG_EXPANSION_FLAG_NAME=TRUE`.
Reviewed By: michaelrj, sivachandra
Differential Revision: https://reviews.llvm.org/D125174
2022-05-09 01:45:40 +08:00
|
|
|
if(SHOW_INTERMEDIATE_OBJECTS)
|
|
|
|
message(STATUS "Adding header library ${fq_target_name}")
|
|
|
|
if(${SHOW_INTERMEDIATE_OBJECTS} STREQUAL "DEPS")
|
|
|
|
foreach(dep IN LISTS ADD_HEADER_DEPENDS)
|
|
|
|
message(STATUS " ${fq_target_name} depends on ${dep}")
|
|
|
|
endforeach()
|
|
|
|
endif()
|
|
|
|
endif()
|
2020-04-22 01:10:58 +08:00
|
|
|
set(interface_target_name "${fq_target_name}.__header_library__")
|
2020-04-18 09:42:40 +08:00
|
|
|
|
|
|
|
add_library(${interface_target_name} INTERFACE)
|
|
|
|
target_sources(${interface_target_name} INTERFACE ${FULL_HDR_PATHS})
|
|
|
|
if(ADD_HEADER_DEPENDS)
|
[libc] Implement FLAGS option for generating all combinations for targets.
Add FLAGS option for add_header_library, add_object_library,
add_entrypoint_object, and add_libc_unittest.
In general, a flag is a string provided for supported functions under the
multi-valued option `FLAGS`. It should be one of the following forms:
FLAG_NAME
FLAG_NAME__NO
FLAG_NAME__ONLY
A target will inherit all the flags of its upstream dependency.
When we create a target `TARGET_NAME` with a flag using (add_header_library,
add_object_library, ...), its behavior will depend on the flag form as follow:
- FLAG_NAME: The following 2 targets will be generated:
`TARGET_NAME` that has `FLAG_NAME` in its `FLAGS` property.
`TARGET_NAME.__NO_FLAG_NAME` that depends on `DEP.__NO_FLAG_NAME` if
`TARGET_NAME` depends on `DEP` and `DEP` has `FLAG_NAME` in its `FLAGS`
property.
- FLAG_NAME__ONLY: Only generate 1 target `TARGET_NAME` that has `FLAG_NAME`
in its `FLAGS` property.
- FLAG_NAME__NO: Only generate 1 target `TARGET_NAME.__NO_FLAG_NAME` that
depends on `DEP.__NO_FLAG_NAME` if `DEP` is in its DEPENDS list and `DEP`
has `FLAG_NAME` in its `FLAGS` property.
To show all the targets generated, pass SHOW_INTERMEDIATE_OBJECTS=ON to cmake.
To show all the targets' dependency and flags, pass
`SHOW_INTERMEDIATE_OBJECTS=DEPS` to cmake.
To completely disable a flag FLAG_NAME expansion, set the variable
`SKIP_FLAG_EXPANSION_FLAG_NAME=TRUE`.
Reviewed By: michaelrj, sivachandra
Differential Revision: https://reviews.llvm.org/D125174
2022-05-09 01:45:40 +08:00
|
|
|
add_dependencies(${interface_target_name} ${ADD_HEADER_DEPENDS})
|
2020-04-18 09:42:40 +08:00
|
|
|
endif()
|
[libc] Implement FLAGS option for generating all combinations for targets.
Add FLAGS option for add_header_library, add_object_library,
add_entrypoint_object, and add_libc_unittest.
In general, a flag is a string provided for supported functions under the
multi-valued option `FLAGS`. It should be one of the following forms:
FLAG_NAME
FLAG_NAME__NO
FLAG_NAME__ONLY
A target will inherit all the flags of its upstream dependency.
When we create a target `TARGET_NAME` with a flag using (add_header_library,
add_object_library, ...), its behavior will depend on the flag form as follow:
- FLAG_NAME: The following 2 targets will be generated:
`TARGET_NAME` that has `FLAG_NAME` in its `FLAGS` property.
`TARGET_NAME.__NO_FLAG_NAME` that depends on `DEP.__NO_FLAG_NAME` if
`TARGET_NAME` depends on `DEP` and `DEP` has `FLAG_NAME` in its `FLAGS`
property.
- FLAG_NAME__ONLY: Only generate 1 target `TARGET_NAME` that has `FLAG_NAME`
in its `FLAGS` property.
- FLAG_NAME__NO: Only generate 1 target `TARGET_NAME.__NO_FLAG_NAME` that
depends on `DEP.__NO_FLAG_NAME` if `DEP` is in its DEPENDS list and `DEP`
has `FLAG_NAME` in its `FLAGS` property.
To show all the targets generated, pass SHOW_INTERMEDIATE_OBJECTS=ON to cmake.
To show all the targets' dependency and flags, pass
`SHOW_INTERMEDIATE_OBJECTS=DEPS` to cmake.
To completely disable a flag FLAG_NAME expansion, set the variable
`SKIP_FLAG_EXPANSION_FLAG_NAME=TRUE`.
Reviewed By: michaelrj, sivachandra
Differential Revision: https://reviews.llvm.org/D125174
2022-05-09 01:45:40 +08:00
|
|
|
set_target_properties(
|
|
|
|
${interface_target_name}
|
|
|
|
PROPERTIES
|
|
|
|
INTERFACE_FLAGS "${ADD_HEADER_FLAGS}"
|
|
|
|
)
|
2020-04-18 09:42:40 +08:00
|
|
|
|
|
|
|
add_custom_target(${fq_target_name})
|
|
|
|
add_dependencies(${fq_target_name} ${interface_target_name})
|
|
|
|
set_target_properties(
|
|
|
|
${fq_target_name}
|
|
|
|
PROPERTIES
|
[libc] Implement FLAGS option for generating all combinations for targets.
Add FLAGS option for add_header_library, add_object_library,
add_entrypoint_object, and add_libc_unittest.
In general, a flag is a string provided for supported functions under the
multi-valued option `FLAGS`. It should be one of the following forms:
FLAG_NAME
FLAG_NAME__NO
FLAG_NAME__ONLY
A target will inherit all the flags of its upstream dependency.
When we create a target `TARGET_NAME` with a flag using (add_header_library,
add_object_library, ...), its behavior will depend on the flag form as follow:
- FLAG_NAME: The following 2 targets will be generated:
`TARGET_NAME` that has `FLAG_NAME` in its `FLAGS` property.
`TARGET_NAME.__NO_FLAG_NAME` that depends on `DEP.__NO_FLAG_NAME` if
`TARGET_NAME` depends on `DEP` and `DEP` has `FLAG_NAME` in its `FLAGS`
property.
- FLAG_NAME__ONLY: Only generate 1 target `TARGET_NAME` that has `FLAG_NAME`
in its `FLAGS` property.
- FLAG_NAME__NO: Only generate 1 target `TARGET_NAME.__NO_FLAG_NAME` that
depends on `DEP.__NO_FLAG_NAME` if `DEP` is in its DEPENDS list and `DEP`
has `FLAG_NAME` in its `FLAGS` property.
To show all the targets generated, pass SHOW_INTERMEDIATE_OBJECTS=ON to cmake.
To show all the targets' dependency and flags, pass
`SHOW_INTERMEDIATE_OBJECTS=DEPS` to cmake.
To completely disable a flag FLAG_NAME expansion, set the variable
`SKIP_FLAG_EXPANSION_FLAG_NAME=TRUE`.
Reviewed By: michaelrj, sivachandra
Differential Revision: https://reviews.llvm.org/D125174
2022-05-09 01:45:40 +08:00
|
|
|
TARGET_TYPE "${HDR_LIBRARY_TARGET_TYPE}"
|
|
|
|
DEPS "${ADD_HEADER_DEPENDS}"
|
|
|
|
FLAGS "${ADD_HEADER_FLAGS}"
|
|
|
|
)
|
|
|
|
endfunction(create_header_library)
|
|
|
|
|
|
|
|
# Rule to add header only libraries.
|
|
|
|
# Usage
|
|
|
|
# add_header_library(
|
|
|
|
# <target name>
|
|
|
|
# HDRS <list of .h files part of the library>
|
|
|
|
# DEPENDS <list of dependencies>
|
|
|
|
# FLAGS <list of flags>
|
|
|
|
# )
|
|
|
|
|
|
|
|
# Internal function, used by `add_header_library`.
|
|
|
|
function(expand_flags_for_header_library target_name flags)
|
|
|
|
cmake_parse_arguments(
|
|
|
|
"EXPAND_FLAGS"
|
|
|
|
"IGNORE_MARKER" # Optional arguments
|
|
|
|
"" # Single-value arguments
|
|
|
|
"DEPENDS;FLAGS" # Multi-value arguments
|
|
|
|
${ARGN}
|
|
|
|
)
|
|
|
|
|
|
|
|
list(LENGTH flags nflags)
|
|
|
|
if(NOT ${nflags})
|
|
|
|
create_header_library(
|
|
|
|
${target_name}
|
|
|
|
DEPENDS ${EXPAND_FLAGS_DEPENDS}
|
|
|
|
FLAGS ${EXPAND_FLAGS_FLAGS}
|
|
|
|
${EXPAND_FLAGS_UNPARSED_ARGUMENTS}
|
|
|
|
)
|
|
|
|
return()
|
|
|
|
endif()
|
|
|
|
|
2022-06-07 01:30:13 +08:00
|
|
|
list(GET flags 0 flag)
|
|
|
|
list(REMOVE_AT flags 0)
|
[libc] Implement FLAGS option for generating all combinations for targets.
Add FLAGS option for add_header_library, add_object_library,
add_entrypoint_object, and add_libc_unittest.
In general, a flag is a string provided for supported functions under the
multi-valued option `FLAGS`. It should be one of the following forms:
FLAG_NAME
FLAG_NAME__NO
FLAG_NAME__ONLY
A target will inherit all the flags of its upstream dependency.
When we create a target `TARGET_NAME` with a flag using (add_header_library,
add_object_library, ...), its behavior will depend on the flag form as follow:
- FLAG_NAME: The following 2 targets will be generated:
`TARGET_NAME` that has `FLAG_NAME` in its `FLAGS` property.
`TARGET_NAME.__NO_FLAG_NAME` that depends on `DEP.__NO_FLAG_NAME` if
`TARGET_NAME` depends on `DEP` and `DEP` has `FLAG_NAME` in its `FLAGS`
property.
- FLAG_NAME__ONLY: Only generate 1 target `TARGET_NAME` that has `FLAG_NAME`
in its `FLAGS` property.
- FLAG_NAME__NO: Only generate 1 target `TARGET_NAME.__NO_FLAG_NAME` that
depends on `DEP.__NO_FLAG_NAME` if `DEP` is in its DEPENDS list and `DEP`
has `FLAG_NAME` in its `FLAGS` property.
To show all the targets generated, pass SHOW_INTERMEDIATE_OBJECTS=ON to cmake.
To show all the targets' dependency and flags, pass
`SHOW_INTERMEDIATE_OBJECTS=DEPS` to cmake.
To completely disable a flag FLAG_NAME expansion, set the variable
`SKIP_FLAG_EXPANSION_FLAG_NAME=TRUE`.
Reviewed By: michaelrj, sivachandra
Differential Revision: https://reviews.llvm.org/D125174
2022-05-09 01:45:40 +08:00
|
|
|
extract_flag_modifier(${flag} real_flag modifier)
|
|
|
|
|
|
|
|
if(NOT "${modifier}" STREQUAL "NO")
|
|
|
|
expand_flags_for_header_library(
|
|
|
|
${target_name}
|
|
|
|
"${flags}"
|
|
|
|
DEPENDS ${EXPAND_FLAGS_DEPENDS} IGNORE_MARKER
|
|
|
|
FLAGS ${EXPAND_FLAGS_FLAGS} IGNORE_MARKER
|
|
|
|
${EXPAND_FLAGS_UNPARSED_ARGUMENTS}
|
|
|
|
)
|
|
|
|
endif()
|
|
|
|
|
|
|
|
if("${real_flag}" STREQUAL "" OR "${modifier}" STREQUAL "ONLY")
|
|
|
|
return()
|
|
|
|
endif()
|
|
|
|
|
|
|
|
set(NEW_FLAGS ${EXPAND_FLAGS_FLAGS})
|
|
|
|
list(REMOVE_ITEM NEW_FLAGS ${flag})
|
|
|
|
get_fq_dep_list_without_flag(NEW_DEPS ${real_flag} ${EXPAND_FLAGS_DEPENDS})
|
|
|
|
|
|
|
|
# Only target with `flag` has `.__NO_flag` target, `flag__NO` and
|
|
|
|
# `flag__ONLY` do not.
|
|
|
|
if(NOT "${modifier}")
|
|
|
|
set(TARGET_NAME "${target_name}.__NO_${flag}")
|
|
|
|
else()
|
|
|
|
set(TARGET_NAME "${target_name}")
|
|
|
|
endif()
|
|
|
|
|
|
|
|
expand_flags_for_header_library(
|
|
|
|
${TARGET_NAME}
|
|
|
|
"${flags}"
|
|
|
|
DEPENDS ${NEW_DEPS} IGNORE_MARKER
|
|
|
|
FLAGS ${NEW_FLAGS} IGNORE_MARKER
|
|
|
|
${EXPAND_FLAGS_UNPARSED_ARGUMENTS}
|
|
|
|
)
|
|
|
|
endfunction(expand_flags_for_header_library)
|
|
|
|
|
|
|
|
function(add_header_library target_name)
|
|
|
|
cmake_parse_arguments(
|
|
|
|
"ADD_TO_EXPAND"
|
|
|
|
"" # Optional arguments
|
|
|
|
"" # Single value arguments
|
|
|
|
"DEPENDS;FLAGS" # Multi-value arguments
|
|
|
|
${ARGN}
|
|
|
|
)
|
|
|
|
|
|
|
|
get_fq_target_name(${target_name} fq_target_name)
|
|
|
|
|
|
|
|
if(ADD_TO_EXPAND_DEPENDS AND ("${SHOW_INTERMEDIATE_OBJECTS}" STREQUAL "DEPS"))
|
|
|
|
message(STATUS "Gathering FLAGS from dependencies for ${fq_target_name}")
|
|
|
|
endif()
|
|
|
|
|
|
|
|
get_fq_deps_list(fq_deps_list ${ADD_TO_EXPAND_DEPENDS})
|
|
|
|
get_flags_from_dep_list(deps_flag_list ${fq_deps_list})
|
|
|
|
|
|
|
|
list(APPEND ADD_TO_EXPAND_FLAGS ${deps_flag_list})
|
|
|
|
remove_duplicated_flags("${ADD_TO_EXPAND_FLAGS}" flags)
|
|
|
|
list(SORT flags)
|
|
|
|
|
|
|
|
if(SHOW_INTERMEDIATE_OBJECTS AND flags)
|
|
|
|
message(STATUS "Header library ${fq_target_name} has FLAGS: ${flags}")
|
|
|
|
endif()
|
|
|
|
|
|
|
|
expand_flags_for_header_library(
|
|
|
|
${fq_target_name}
|
|
|
|
"${flags}"
|
|
|
|
DEPENDS ${fq_deps_list} IGNORE_MARKER
|
|
|
|
FLAGS ${flags} IGNORE_MARKER
|
|
|
|
${ADD_TO_EXPAND_UNPARSED_ARGUMENTS}
|
2020-04-18 09:42:40 +08:00
|
|
|
)
|
|
|
|
endfunction(add_header_library)
|