aboutsummaryrefslogtreecommitdiff
path: root/modules/ECMGenerateExportHeader.cmake
diff options
context:
space:
mode:
Diffstat (limited to 'modules/ECMGenerateExportHeader.cmake')
-rw-r--r--modules/ECMGenerateExportHeader.cmake729
1 files changed, 729 insertions, 0 deletions
diff --git a/modules/ECMGenerateExportHeader.cmake b/modules/ECMGenerateExportHeader.cmake
new file mode 100644
index 00000000..dfecb3a6
--- /dev/null
+++ b/modules/ECMGenerateExportHeader.cmake
@@ -0,0 +1,729 @@
+#.rst:
+# ECMGenerateExportHeader
+# -----------------------
+#
+# This module provides the ``ecm_generate_export_header`` function for
+# generating export macros for libraries with version-based control over
+# visibility of and compiler warnings for deprecated API for the library user,
+# as well as over excluding deprecated API and their implementation when
+# building the library itself.
+#
+# For preparing some values useful in the context it also provides a function
+# ``ecm_export_header_format_version``.
+#
+# ::
+#
+# ecm_generate_export_header(<library_target_name>
+# VERSION <version>
+# [BASE_NAME <base_name>]
+# [GROUP_BASE_NAME <group_base_name>]
+# [EXPORT_MACRO_NAME <export_macro_name>]
+# [EXPORT_FILE_NAME <export_file_name>]
+# [DEPRECATED_MACRO_NAME <deprecated_macro_name>]
+# [NO_EXPORT_MACRO_NAME <no_export_macro_name>]
+# [INCLUDE_GUARD_NAME <include_guard_name>]
+# [STATIC_DEFINE <static_define>]
+# [PREFIX_NAME <prefix_name>]
+# [DEPRECATED_BASE_VERSION <deprecated_base_version>]
+# [DEPRECATION_VERSIONS <deprecation_version> [<deprecation_version2> [...]]]
+# [EXCLUDE_DEPRECATED_BEFORE_AND_AT <exclude_deprecated_before_and_at_version>]
+# [NO_DEFINITION_EXPORT_TO_BUILD_INTERFACE]
+# [CUSTOM_CONTENT_FROM_VARIABLE <variable>]
+# )
+#
+# ``VERSION`` specifies the version of the library, given in the format
+# "<major>.<minor>.<patchlevel>".
+#
+# ``GROUP_BASE_NAME`` specifies the name to use for the macros defining
+# library group default values. If set, this will generate code supporting
+# ``<group_base_name>_NO_DEPRECATED_WARNINGS``,
+# ``<group_base_name>_DISABLE_DEPRECATED_BEFORE_AND_AT``,
+# ``<group_base_name>_DEPRECATED_WARNINGS_SINCE`` and
+# ``<group_base_name>_NO_DEPRECATED`` (see below).
+# If not set, the generated code will ignore any such macros.
+#
+# ``DEPRECATED_BASE_VERSION`` specifies the default version for warnings
+# about API deprecated before and at. The default is the value of
+# "<exclude_deprecated_before_and_at_version>" if set, or "<major>.0.0", with
+# <major> taken from <version>.
+#
+# ``DEPRECATION_VERSIONS`` specifies versions in "<major>.<minor>" format in
+# which API was declared deprecated. Any version used with the generated
+# macro ``<prefix_name><base_name>_DEPRECATED_VERSION(major, minor, text)``
+# needs to be listed here, otherwise the macro will fail to work.
+#
+# ``EXCLUDE_DEPRECATED_BEFORE_AND_AT`` specifies the version for which all API
+# deprecated before and at should be excluded from the build completely.
+# Possible values are "0" (default), "CURRENT" (which resolves to <version>)
+# and a version string in the format "<major>.<minor>.<patchlevel>".
+#
+# ``NO_DEFINITION_EXPORT_TO_BUILD_INTERFACE`` specifies that the definition
+# ``<prefix_name><uppercase_base_name>_DISABLE_DEPRECATED_BEFORE_AND_AT`` will
+# not be set in the public interface of the library inside its own build.
+# The default is that it is set, to the version specified by
+# ``EXCLUDE_DEPRECATED_BEFORE_AND_AT``, so e.g. test and examples part of the
+# project automatically build against the full API included in the build.
+#
+#
+# The function ``ecm_generate_export_header`` defines C++ preprocessor macros
+# in the generated export header, some for use in the sources of the library
+# the header is generated for, other for use by projects linking agsinst the
+# library.
+#
+# The macros for use in the library C++ sources are these, next to those also
+# defined by `GenerateExportHeader
+# <https://cmake.org/cmake/help/latest/module/GenerateExportHeader.html>`_:
+#
+# ``<prefix_name><uppercase_base_name>_DEPRECATED_VERSION(major, minor, text)``
+# to use to conditionally set a
+# ``<prefix_name><uppercase_base_name>_DEPRECATED`` macro for a class, struct
+# or function (other elements to be supported in future versions), depending
+# on the visibility macro flags set (see below)
+#
+# ``<prefix_name><uppercase_base_name>_ENABLE_DEPRECATED_SINCE(major, minor)``
+# evaluates to ``TRUE`` or ``FALSE`` depending on the visibility macro flags
+# set (see below). To be used mainly with ``#if``/``#endif`` to mark sections
+# of code which should be included depending on the visibility requested.
+#
+# ``<prefix_name><uppercase_base_name>_BUILD_DEPRECATED_SINCE(major, minor)``
+# evaluates to ``TRUE`` or ``FALSE`` depending on the value of
+# ``EXCLUDE_DEPRECATED_BEFORE_AND_AT``. To be used mainly with
+# ``#if``/``#endif`` to mark sections of two types of code: implementation
+# code for deprecated API and declaration code of deprecated API which only
+# may be disabled at build time of the library for BC reasons (e.g. virtual
+# methods, see notes below).
+#
+# ``<prefix_name><uppercase_base_name>_EXCLUDE_DEPRECATED_BEFORE_AND_AT``
+# holds the version used to exclude deprecated API at build time of the
+# library.
+#
+# The macros used to control visibility when building against the library are:
+#
+# ``<prefix_name><uppercase_base_name>_DISABLE_DEPRECATED_BEFORE_AND_AT``
+# definition to set to a value in single hex number version notation
+# (``0x<major><minor><patchlevel>``).
+#
+# ``<prefix_name><uppercase_base_name>_NO_DEPRECATED``
+# flag to define to disable all deprecated API, being a shortcut for
+# settings ``<prefix_name><uppercase_base_name>_DISABLE_DEPRECATED_BEFORE_AND_AT``
+# to the current version. If both are set, this flag overrules.
+#
+# ``<prefix_name><uppercase_base_name>_DEPRECATED_WARNINGS_SINCE``
+# definition to set to a value in single hex number version notation
+# (``0x<major><minor><patchlevel>``). Warnings will be only activated for
+# API deprecated up to and including the version. If
+# ``<prefix_name><uppercase_base_name>_DISABLE_DEPRECATED_BEFORE_AND_AT``
+# is set (directly or via the group default), it will default to that
+# version, resulting in no warnings. Otherwise the default is the current
+# version, resulting in warnings for all deprecated API.
+#
+# ``<prefix_name><uppercase_base_name>_NO_DEPRECATED_WARNINGS``
+# flag to define to disable all deprecation warnings, being a shortcut for
+# setting ``<prefix_name><uppercase_base_name>_DEPRECATED_WARNINGS_SINCE``
+# to "0". If both are set, this flag overrules.
+#
+# When the ``GROUP_BASE_NAME`` has been used, the same macros but with the
+# given ``<group_base_name>`` prefix are available to define the defaults of
+# these macros, if not explicitely set.
+#
+# Note: The tricks applied here for hiding deprecated API to the compiler
+# when building against a library do not work for all deprecated API:
+#
+# * virtual methods need to stay visible to the compiler to build proper
+# virtual method tables for subclasses
+# * enumerators from enums cannot be simply removed, as this changes
+# auto values of following enumerators, also can poke holes in enumerator
+# series used as index into tables
+#
+# In such cases the API can be only "hidden" at build time of the library,
+# itself, by generated hard coded macro settings, using
+# ``<prefix_name><uppercase_base_name>_BUILD_DEPRECATED_SINCE(major, minor)``.
+#
+# Examples:
+#
+# Preparing a library "Foo" created by target "foo", which is part of a group
+# of libraries "Bar", where some API of "Foo" got deprecated at versions
+# 5.0 & 5.12:
+#
+# .. code-block:: cmake
+#
+# ecm_generate_export_header(foo
+# GROUP_BASE_NAME BAR
+# VERSION ${FOO_VERSION}
+# DEPRECATION_VERSIONS 5.0 5.12
+# )
+#
+# In the library "Foo" sources in the headers the API would be prepared like
+# this, using the generated macros ``FOO_ENABLE_DEPRECATED_SINCE`` and
+# ``FOO_DEPRECATED_VERSION``:
+#
+# .. code-block:: cpp
+#
+# #include <foo_export.h>
+#
+# #if FOO_ENABLE_DEPRECATED_SINCE(5, 0)
+# /**
+# * @deprecated Since 5.0
+# */
+# FOO_DEPRECATED_VERSION(5, 0, "Use doFoo2()")
+# FOO_EXPORT void doFoo();
+# #endif
+#
+# #if FOO_ENABLE_DEPRECATED_SINCE(5, 12)
+# /**
+# * @deprecated Since 5.12
+# */
+# FOO_DEPRECATED_VERSION(5, 12, "Use doBar2()")
+# FOO_EXPORT void doBar();
+# #endif
+#
+# Projects linking against the "Foo" library can control which part of its
+# deprecated API should be hidden to the compiler by adding a definition
+# using the ``FOO_DISABLE_DEPRECATED_BEFORE_AND_AT`` macro variable set to the
+# desired value (in version hex number notation):
+#
+# .. code-block:: cmake
+#
+# add_definitions(-DFOO_DISABLE_DEPRECATED_BEFORE_AND_AT=0x050000)
+#
+# Or using the macro variable of the group:
+#
+# .. code-block:: cmake
+#
+# add_definitions(-DBAR_DISABLE_DEPRECATED_BEFORE_AND_AT=0x050000)
+#
+# If both are specified, ``FOO_DISABLE_DEPRECATED_BEFORE_AND_AT`` will take
+# precedence.
+#
+# To build a variant of a library with some deprecated API completely left
+# out from the build, not only optionally invisible to consumers, one uses the
+# ``EXCLUDE_DEPRECATED_BEFORE_AND_AT`` parameter. This is best combined with a
+# cached CMake variable.
+#
+# .. code-block:: cmake
+#
+# set(EXCLUDE_DEPRECATED_BEFORE_AND_AT 0 CACHE STRING "Control the range of deprecated API excluded from the build [default=0].")
+#
+# ecm_generate_export_header(foo
+# VERSION ${FOO_VERSION}
+# EXCLUDE_DEPRECATED_BEFORE_AND_AT ${EXCLUDE_DEPRECATED_BEFORE_AND_AT}
+# DEPRECATION_VERSIONS 5.0 5.12
+# )
+#
+# The macros used in the headers for library consumers are reused for
+# disabling the API excluded in the build of the library. For disabling the
+# implementation of that API as well as for disabling deprecated API which
+# only can be disabled at build time of the library for BC reasons, one
+# uses the generated macro ``FOO_BUILD_DEPRECATED_SINCE``, like this:
+#
+# .. code-block:: cpp
+#
+# #include <foo_export.h>
+#
+# enum Bars {
+# One,
+# #if FOO_BUILD_DEPRECATED_SINCE(5, 0)
+# Two,
+# #endif
+# Three,
+# };
+#
+# #if FOO_ENABLE_DEPRECATED_SINCE(5, 0)
+# /**
+# * @deprecated Since 5.0
+# */
+# FOO_DEPRECATED_VERSION(5, 0, "Use doFoo2()")
+# FOO_EXPORT void doFoo();
+# #endif
+#
+# #if FOO_ENABLE_DEPRECATED_SINCE(5, 12)
+# /**
+# * @deprecated Since 5.12
+# */
+# FOO_DEPRECATED_VERSION(5, 12, "Use doBar2()")
+# FOO_EXPORT void doBar();
+# #endif
+#
+# class FOO_EXPORT Foo {
+# public:
+# #if FOO_BUILD_DEPRECATED_SINCE(5, 0)
+# /**
+# * @deprecated Since 5.0
+# */
+# FOO_DEPRECATED_VERSION(5, 0, "Feature removed")
+# virtual void doWhat();
+# #endif
+# };
+#
+# .. code-block:: cpp
+#
+# #if FOO_BUILD_DEPRECATED_SINCE(5, 0)
+# void doFoo()
+# {
+# // [...]
+# }
+# #endif
+#
+# #if FOO_BUILD_DEPRECATED_SINCE(5, 12)
+# void doBar()
+# {
+# // [...]
+# }
+# #endif
+#
+# #if FOO_BUILD_DEPRECATED_SINCE(5, 0)
+# void Foo::doWhat()
+# {
+# // [...]
+# }
+# #endif
+#
+# So e.g. if ``EXCLUDE_DEPRECATED_BEFORE_AND_AT`` is set to "5.0.0", the
+# enumerator ``Two`` as well as the methods ``::doFoo()`` and ``Foo::doWhat()``
+# will be not available to library consumers. The methods will not have been
+# compiled into the library binary, and the declarations will be hidden to the
+# compiler, ``FOO_DISABLE_DEPRECATED_BEFORE_AND_AT`` also cannot be used to
+# reactivate them.
+#
+# When using the ``NO_DEFINITION_EXPORT_TO_BUILD_INTERFACE`` and the project
+# for the "Foo" library includes also tests and examples linking against the
+# library and using deprecated API (like tests covering it), one better
+# explicitely sets ``FOO_DISABLE_DEPRECATED_BEFORE_AND_AT`` for those targets
+# to the version before and at which all deprecated API has been excluded from
+# the build.
+# Even more when building against other libraries from the same group "Bar" and
+# disabling some deprecated API of those libraries using the group macro
+# ``BAR_DISABLE_DEPRECATED_BEFORE_AND_AT``, which also works as default for
+# ``FOO_DISABLE_DEPRECATED_BEFORE_AND_AT``.
+#
+# To get the hex number style value the helper macro
+# ``ecm_export_header_format_version()`` will be used:
+#
+# .. code-block:: cmake
+#
+# set(EXCLUDE_DEPRECATED_BEFORE_AND_AT 0 CACHE STRING "Control what part of deprecated API is excluded from build [default=0].")
+#
+# ecm_generate_export_header(foo
+# VERSION ${FOO_VERSION}
+# GROUP_BASE_NAME BAR
+# EXCLUDE_DEPRECATED_BEFORE_AND_AT ${EXCLUDE_DEPRECATED_BEFORE_AND_AT}
+# NO_DEFINITION_EXPORT_TO_BUILD_INTERFACE
+# DEPRECATION_VERSIONS 5.0 5.12
+# )
+#
+# ecm_export_header_format_version(${EXCLUDE_DEPRECATED_BEFORE_AND_AT}
+# CURRENT_VERSION ${FOO_VERSION}
+# HEXNUMBER_VAR foo_no_deprecated_before_and_at
+# )
+#
+# # disable all deprecated API up to 5.9.0 from all other libs of group "BAR" that we use ourselves
+# add_definitions(-DBAR_DISABLE_DEPRECATED_BEFORE_AND_AT=0x050900)
+#
+# add_executable(app app.cpp)
+# target_link_libraries(app foo)
+# target_compile_definitions(app
+# PRIVATE "FOO_DISABLE_DEPRECATED_BEFORE_AND_AT=${foo_no_deprecated_before_and_at}")
+#
+# Since 5.64.0.
+
+#=============================================================================
+# Copyright 2019 Friedrich W. H. Kossebau <kossebau@kde.org>
+#
+# Redistribution and use in source and binary forms, with or without
+# modification, are permitted provided that the following conditions
+# are met:
+#
+# 1. Redistributions of source code must retain the copyright
+# notice, this list of conditions and the following disclaimer.
+# 2. Redistributions in binary form must reproduce the copyright
+# notice, this list of conditions and the following disclaimer in the
+# documentation and/or other materials provided with the distribution.
+# 3. The name of the author may not be used to endorse or promote products
+# derived from this software without specific prior written permission.
+#
+# THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
+# IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
+# OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
+# IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
+# INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
+# NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+# DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+# THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
+# THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+
+include(GenerateExportHeader)
+include(CMakeParseArguments)
+
+# helper method
+function(_ecm_geh_generate_hex_number _var_name _version)
+ set(_hexnumber 0)
+
+ set(version_regex "^([0-9]+)\\.([0-9]+)\\.([0-9]+)$")
+ string(REGEX REPLACE ${version_regex} "\\1" _version_major "${_version}")
+ string(REGEX REPLACE ${version_regex} "\\2" _version_minor "${_version}")
+ string(REGEX REPLACE ${version_regex} "\\3" _version_patch "${_version}")
+ set(_outputformat)
+ if (NOT CMAKE_VERSION VERSION_LESS 3.13)
+ set(_outputformat OUTPUT_FORMAT HEXADECIMAL)
+ endif()
+ math(EXPR _hexnumber "${_version_major}*65536 + ${_version_minor}*256 + ${_version_patch}" ${_outputformat})
+ set(${_var_name} ${_hexnumber} PARENT_SCOPE)
+endfunction()
+
+function(ecm_export_header_format_version _version)
+ set(options
+ )
+ set(oneValueArgs
+ CURRENT_VERSION
+ STRING_VAR
+ HEXNUMBER_VAR
+ )
+ set(multiValueArgs
+ )
+ cmake_parse_arguments(ARGS "${options}" "${oneValueArgs}" "${multiValueArgs}" ${ARGN})
+
+ if (NOT ARGS_CURRENT_VERSION)
+ message(FATAL_ERROR "No CURRENT_VERSION passed when calling ecm_export_header_format_version().")
+ endif()
+ if (NOT ARGS_STRING_VAR AND NOT ARGS_HEXNUMBER_VAR)
+ message(FATAL_ERROR "No STRING_VAR or HEXNUMBER_VAR passed when calling ecm_export_header_format_version().")
+ endif()
+
+ if(_version STREQUAL "CURRENT")
+ set(_version ${ARGS_CURRENT_VERSION})
+ endif()
+
+ if (ARGS_STRING_VAR)
+ set(${ARGS_STRING_VAR} ${_version} PARENT_SCOPE)
+ endif()
+
+ if (ARGS_HEXNUMBER_VAR)
+ set(_hexnumber 0) # by default build all deprecated API
+ if(_version)
+ _ecm_geh_generate_hex_number(_hexnumber ${_version})
+ endif()
+ set(${ARGS_HEXNUMBER_VAR} ${_hexnumber} PARENT_SCOPE)
+ endif()
+endfunction()
+
+
+function(ecm_generate_export_header target)
+ set(options
+ NO_DEFINITION_EXPORT_TO_BUILD_INTERFACE
+ )
+ set(oneValueArgs
+ BASE_NAME
+ GROUP_BASE_NAME
+ EXPORT_FILE_NAME
+ DEPRECATED_BASE_VERSION
+ VERSION
+ EXCLUDE_DEPRECATED_BEFORE_AND_AT
+ EXPORT_MACRO_NAME
+ DEPRECATED_MACRO_NAME
+ NO_EXPORT_MACRO_NAME
+ INCLUDE_GUARD_NAME
+ STATIC_DEFINE
+ PREFIX_NAME
+ CUSTOM_CONTENT_FROM_VARIABLE
+ )
+ set(multiValueArgs
+ DEPRECATION_VERSIONS
+ )
+ cmake_parse_arguments(ARGS "${options}" "${oneValueArgs}" "${multiValueArgs}" ${ARGN})
+
+ # args sanity check
+ if (NOT ARGS_VERSION)
+ message(FATAL_ERROR "No VERSION passed when calling ecm_generate_export_header().")
+ endif()
+ if (ARGS_INCLUDE_GUARD_NAME AND CMAKE_VERSION VERSION_LESS 3.11)
+ message(FATAL_ERROR "Argument INCLUDE_GUARD_NAME needs at least CMake 3.11 when calling ecm_generate_export_header().")
+ endif()
+ if (NOT ARGS_EXCLUDE_DEPRECATED_BEFORE_AND_AT)
+ set(ARGS_EXCLUDE_DEPRECATED_BEFORE_AND_AT 0)
+ elseif(ARGS_EXCLUDE_DEPRECATED_BEFORE_AND_AT STREQUAL "CURRENT")
+ set(ARGS_EXCLUDE_DEPRECATED_BEFORE_AND_AT ${ARGS_VERSION})
+ endif()
+ if (NOT ARGS_DEPRECATED_BASE_VERSION)
+ if (ARGS_EXCLUDE_DEPRECATED_BEFORE_AND_AT)
+ set(ARGS_DEPRECATED_BASE_VERSION "${ARGS_EXCLUDE_DEPRECATED_BEFORE_AND_AT}")
+ else()
+ string(REGEX REPLACE "^([0-9]+)\\.([0-9]+)\\.([0-9]+)$" "\\1"
+ _version_major "${ARGS_VERSION}")
+ set(ARGS_DEPRECATED_BASE_VERSION "${_version_major}.0.0")
+ endif()
+ else()
+ if (ARGS_DEPRECATED_BASE_VERSION VERSION_LESS ARGS_EXCLUDE_DEPRECATED_BEFORE_AND_AT)
+ message(FATAL_ERROR "DEPRECATED_BASE_VERSION cannot be lower than EXCLUDE_DEPRECATED_BEFORE_AND_AT when calling ecm_generate_export_header().")
+ endif()
+ endif()
+ if(NOT ARGS_BASE_NAME)
+ set(ARGS_BASE_NAME "${target}")
+ endif()
+ string(TOUPPER "${ARGS_BASE_NAME}" _upper_base_name)
+ string(TOLOWER "${ARGS_BASE_NAME}" _lower_base_name)
+
+ if(NOT ARGS_EXPORT_FILE_NAME)
+ set(ARGS_EXPORT_FILE_NAME "${_lower_base_name}_export.h")
+ endif()
+ string(REPLACE "." ", " _commaed_version "${ARGS_VERSION}")
+ string(REPLACE "." ", " _commaed_deprecatedbaseversion "${ARGS_DEPRECATED_BASE_VERSION}")
+
+ set(_macro_base_name "${ARGS_PREFIX_NAME}${_upper_base_name}")
+ if (ARGS_EXPORT_MACRO_NAME)
+ set(_export_macro_name "${ARGS_EXPORT_MACRO_NAME}")
+ else()
+ set(_export_macro_name "${_macro_base_name}_EXPORT")
+ endif()
+ if (ARGS_NO_EXPORT_MACRO_NAME)
+ set(_no_export_macro_name "${ARGS_NO_EXPORT_MACRO_NAME}")
+ else()
+ set(_no_export_macro_name "${_macro_base_name}_NO_EXPORT")
+ endif()
+ if (ARGS_DEPRECATED_MACRO_NAME)
+ set(_deprecated_macro_name "${ARGS_DEPRECATED_MACRO_NAME}")
+ else()
+ set(_deprecated_macro_name "${_macro_base_name}_DEPRECATED")
+ endif()
+
+ if(NOT IS_ABSOLUTE ${ARGS_EXPORT_FILE_NAME})
+ set(ARGS_EXPORT_FILE_NAME "${CMAKE_CURRENT_BINARY_DIR}/${ARGS_EXPORT_FILE_NAME}")
+ endif()
+
+ set_property(TARGET ${target} APPEND PROPERTY AUTOGEN_TARGET_DEPENDS "${ARGS_EXPORT_FILE_NAME}")
+ # build with all the API not excluded, ensure all deprecated API is visible in the build itselt
+ _ecm_geh_generate_hex_number(_hexnumber ${ARGS_EXCLUDE_DEPRECATED_BEFORE_AND_AT})
+ set(_disabling_definition "${_macro_base_name}_DISABLE_DEPRECATED_BEFORE_AND_AT=${_hexnumber}")
+ target_compile_definitions(${target} PRIVATE "${_disabling_definition}")
+ if (NOT ARGS_NO_DEFINITION_EXPORT_TO_BUILD_INTERFACE)
+ target_compile_definitions(${target} INTERFACE "$<BUILD_INTERFACE:${_disabling_definition}>")
+ endif()
+
+ # for the set of compiler versions supported by ECM/KF we can assume those attributes supported
+ # KF6: with C++14 support expected, switch to always use [[deprecated(text)]]
+ if (CMAKE_C_COMPILER_ID STREQUAL "GNU" OR CMAKE_C_COMPILER_ID MATCHES "Clang")
+ set(_decl_deprecated_text_definition "__attribute__ ((__deprecated__(text)))")
+ elseif(MSVC)
+ set(_decl_deprecated_text_definition "__declspec(deprecated(text))")
+ else()
+ set(_decl_deprecated_text_definition "${_macro_base_name}_DECL_DEPRECATED")
+ endif()
+
+ # generate header file
+ set(_output "
+#define ${_macro_base_name}_DECL_DEPRECATED_TEXT(text) ${_decl_deprecated_text_definition}
+
+#define ECM_GENERATEEXPORTHEADER_VERSION_VALUE(major, minor, patch) ((major<<16)|(minor<<8)|(patch))
+"
+ )
+ if (ARGS_GROUP_BASE_NAME)
+ string(TOUPPER "${ARGS_GROUP_BASE_NAME}" _upper_group_name)
+ string(APPEND _output "
+// Take any defaults from group settings
+#if !defined(${_macro_base_name}_NO_DEPRECATED) && !defined(${_macro_base_name}_DISABLE_DEPRECATED_BEFORE_AND_AT)
+# ifdef ${_upper_group_name}_NO_DEPRECATED
+# define ${_macro_base_name}_NO_DEPRECATED
+# elif defined(${_upper_group_name}_DISABLE_DEPRECATED_BEFORE_AND_AT)
+# define ${_macro_base_name}_DISABLE_DEPRECATED_BEFORE_AND_AT ${_upper_group_name}_DISABLE_DEPRECATED_BEFORE_AND_AT
+# endif
+#endif
+#if !defined(${_macro_base_name}_DISABLE_DEPRECATED_BEFORE_AND_AT) && defined(${_upper_group_name}_DISABLE_DEPRECATED_BEFORE_AND_AT)
+# define ${_macro_base_name}_DISABLE_DEPRECATED_BEFORE_AND_AT ${_upper_group_name}_DISABLE_DEPRECATED_BEFORE_AND_AT
+#endif
+
+#if !defined(${_macro_base_name}_NO_DEPRECATED_WARNINGS) && !defined(${_macro_base_name}_DEPRECATED_WARNINGS_SINCE)
+# ifdef ${_upper_group_name}_NO_DEPRECATED_WARNINGS
+# define ${_macro_base_name}_NO_DEPRECATED_WARNINGS
+# elif defined(${_upper_group_name}_DEPRECATED_WARNINGS_SINCE)
+# define ${_macro_base_name}_DEPRECATED_WARNINGS_SINCE ${_upper_group_name}_DEPRECATED_WARNINGS_SINCE
+# endif
+#endif
+#if !defined(${_macro_base_name}_DEPRECATED_WARNINGS_SINCE) && defined(${_upper_group_name}_DEPRECATED_WARNINGS_SINCE)
+# define ${_macro_base_name}_DEPRECATED_WARNINGS_SINCE ${_upper_group_name}_DEPRECATED_WARNINGS_SINCE
+#endif
+"
+ )
+ endif()
+ string(APPEND _output "
+#if defined(${_macro_base_name}_NO_DEPRECATED)
+# undef ${_deprecated_macro_name}
+# define ${_deprecated_macro_name}_EXPORT ${_export_macro_name}
+# define ${_deprecated_macro_name}_NO_EXPORT ${_no_export_macro_name}
+#elif defined(${_macro_base_name}_NO_DEPRECATED_WARNINGS)
+# define ${_deprecated_macro_name}
+# define ${_deprecated_macro_name}_EXPORT ${_export_macro_name}
+# define ${_deprecated_macro_name}_NO_EXPORT ${_no_export_macro_name}
+#else
+# define ${_deprecated_macro_name} ${_macro_base_name}_DECL_DEPRECATED
+# define ${_deprecated_macro_name}_EXPORT ${_macro_base_name}_DECL_DEPRECATED_EXPORT
+# define ${_deprecated_macro_name}_NO_EXPORT ${_macro_base_name}_DECL_DEPRECATED_NO_EXPORT
+#endif
+"
+ )
+ if (ARGS_EXCLUDE_DEPRECATED_BEFORE_AND_AT)
+ message(STATUS "Excluding from build all API deprecated before and at: ${ARGS_EXCLUDE_DEPRECATED_BEFORE_AND_AT}")
+ # TODO: the generated code ideally would emit a warning if some consumer used a value
+ # smaller then what the the build was done with
+ string(REPLACE "." ", " _commaed_excluded_before_version "${ARGS_EXCLUDE_DEPRECATED_BEFORE_AND_AT}")
+ string(APPEND _output "
+// Build was done with the API removed deprecated before: ${ARGS_EXCLUDE_DEPRECATED_BEFORE_AND_AT}
+#define ${_macro_base_name}_EXCLUDE_DEPRECATED_BEFORE_AND_AT ECM_GENERATEEXPORTHEADER_VERSION_VALUE(${_commaed_excluded_before_version})
+
+#ifdef ${_macro_base_name}_DISABLE_DEPRECATED_BEFORE_AND_AT
+# if ${_macro_base_name}_DISABLE_DEPRECATED_BEFORE_AND_AT < ${_macro_base_name}_EXCLUDE_DEPRECATED_BEFORE_AND_AT
+# undef ${_macro_base_name}_DISABLE_DEPRECATED_BEFORE_AND_AT
+# define ${_macro_base_name}_DISABLE_DEPRECATED_BEFORE_AND_AT ${_macro_base_name}_EXCLUDE_DEPRECATED_BEFORE_AND_AT
+# endif
+#endif
+
+#define ${_macro_base_name}_BUILD_DEPRECATED_SINCE(major, minor) (ECM_GENERATEEXPORTHEADER_VERSION_VALUE(major, minor, 0) > ${_macro_base_name}_EXCLUDE_DEPRECATED_BEFORE_AND_AT)
+"
+ )
+ else()
+ string(APPEND _output "
+// No deprecated API had been removed from build
+#define ${_macro_base_name}_EXCLUDE_DEPRECATED_BEFORE_AND_AT 0
+
+#define ${_macro_base_name}_BUILD_DEPRECATED_SINCE(major, minor) 1
+"
+ )
+ endif()
+ string(APPEND _output "
+#ifdef ${_macro_base_name}_NO_DEPRECATED
+# define ${_macro_base_name}_DISABLE_DEPRECATED_BEFORE_AND_AT ECM_GENERATEEXPORTHEADER_VERSION_VALUE(${_commaed_version})
+#endif
+#ifdef ${_macro_base_name}_NO_DEPRECATED_WARNINGS
+# define ${_macro_base_name}_DEPRECATED_WARNINGS_SINCE 0
+#endif
+
+#ifndef ${_macro_base_name}_DEPRECATED_WARNINGS_SINCE
+# ifdef ${_macro_base_name}_DISABLE_DEPRECATED_BEFORE_AND_AT
+# define ${_macro_base_name}_DEPRECATED_WARNINGS_SINCE ${_macro_base_name}_DISABLE_DEPRECATED_BEFORE_AND_AT
+# else
+# define ${_macro_base_name}_DEPRECATED_WARNINGS_SINCE ECM_GENERATEEXPORTHEADER_VERSION_VALUE(${_commaed_version})
+# endif
+#endif
+
+#ifndef ${_macro_base_name}_DISABLE_DEPRECATED_BEFORE_AND_AT
+# define ${_macro_base_name}_DISABLE_DEPRECATED_BEFORE_AND_AT ECM_GENERATEEXPORTHEADER_VERSION_VALUE(${_commaed_deprecatedbaseversion})
+#endif
+
+#ifdef ${_deprecated_macro_name}
+# define ${_macro_base_name}_ENABLE_DEPRECATED_SINCE(major, minor) (ECM_GENERATEEXPORTHEADER_VERSION_VALUE(major, minor, 0) > ${_macro_base_name}_DISABLE_DEPRECATED_BEFORE_AND_AT)
+#else
+# define ${_macro_base_name}_ENABLE_DEPRECATED_SINCE(major, minor) 0
+#endif
+"
+ )
+ if (DEFINED ARGS_DEPRECATION_VERSIONS)
+ set(_major_versions)
+ foreach(_version ${ARGS_DEPRECATION_VERSIONS})
+ string(REPLACE "." ", " _commaed_version "${_version}")
+ string(REPLACE "." "_" _underscored_version "${_version}")
+ string(REGEX REPLACE "^([0-9]+)\\.([0-9]+)$" "\\1"
+ _version_major "${_version}")
+ # IN_LIST only since cmake 3.3
+ set(_in_list FALSE)
+ foreach(_v ${_major_versions})
+ if (_v STREQUAL _version_major)
+ set(_in_list TRUE)
+ break()
+ endif()
+ endforeach()
+ if(NOT _in_list)
+ list(APPEND _major_versions ${_version_major})
+ endif()
+
+ string(APPEND _output "
+#if ${_macro_base_name}_DEPRECATED_WARNINGS_SINCE >= ECM_GENERATEEXPORTHEADER_VERSION_VALUE(${_commaed_version}, 0)
+# define ${_macro_base_name}_DEPRECATED_VERSION_${_underscored_version}(text) ${_macro_base_name}_DECL_DEPRECATED_TEXT(text)
+#else
+# define ${_macro_base_name}_DEPRECATED_VERSION_${_underscored_version}(text)
+#endif
+"
+ )
+ endforeach()
+ foreach(_major_version ${_major_versions})
+ string(APPEND _output
+"#define ${_macro_base_name}_DEPRECATED_VERSION_${_major_version}(minor, text) ${_macro_base_name}_DEPRECATED_VERSION_${_major_version}_##minor(text)
+"
+ )
+ endforeach()
+
+ # TODO: any way to prefix "Since major.minor. " to text without getting text context wrapped with ""?
+ string(APPEND _output
+"#define ${_macro_base_name}_DEPRECATED_VERSION(major, minor, text) ${_macro_base_name}_DEPRECATED_VERSION_##major(minor, \"Since \"#major\".\"#minor\". \"#text)
+"
+ )
+ endif()
+ if (ARGS_CUSTOM_CONTENT_FROM_VARIABLE)
+ string(APPEND _output "${ARGS_CUSTOM_CONTENT_FROM_VARIABLE}\n")
+ endif()
+
+ set(_header_file "${ARGS_EXPORT_FILE_NAME}")
+ set(_header_work_file "${_header_file}.work")
+
+ # prepare optional arguments to pass through to generate_export_header
+ set(_include_guard_name_args)
+ if (ARGS_INCLUDE_GUARD_NAME)
+ set(_include_guard_name_args INCLUDE_GUARD_NAME "${ARGS_INCLUDE_GUARD_NAME}")
+ endif()
+ set(_export_macro_name_args)
+ if (ARGS_EXPORT_MACRO_NAME)
+ set(_export_macro_name_args EXPORT_MACRO_NAME "${ARGS_EXPORT_MACRO_NAME}")
+ endif()
+ set(_no_export_macro_name_args)
+ if (ARGS_NO_EXPORT_MACRO_NAME)
+ set(_no_export_macro_name_args ARGS_NO_EXPORT_MACRO_NAME "${ARGS_NO_EXPORT_MACRO_NAME}")
+ endif()
+ set(_prefix_name_args)
+ if (ARGS_PREFIX_NAME)
+ set(_prefix_name_args PREFIX_NAME "${ARGS_PREFIX_NAME}")
+ endif()
+ set(_static_define_args)
+ if (ARGS_STATIC_DEFINE)
+ set(_static_define_args STATIC_DEFINE "${ARGS_STATIC_DEFINE}")
+ endif()
+ # for older cmake verions we have to manually append our generated content
+ # for newer we use CUSTOM_CONTENT_FROM_VARIABLE
+ set(_custom_content_args)
+ if (NOT CMAKE_VERSION VERSION_LESS 3.7)
+ set(_custom_content_args CUSTOM_CONTENT_FROM_VARIABLE _output)
+ endif()
+ generate_export_header(${target}
+ BASE_NAME ${ARGS_BASE_NAME}
+ DEPRECATED_MACRO_NAME "${_macro_base_name}_DECL_DEPRECATED"
+ ${_prefix_name_args}
+ ${_export_macro_name_args}
+ ${_no_export_macro_name_args}
+ ${_static_define_args}
+ EXPORT_FILE_NAME "${_header_work_file}"
+ ${_include_guard_name_args}
+ ${_custom_content_args}
+ )
+
+ if (CMAKE_VERSION VERSION_LESS 3.7)
+ if (ARGS_INCLUDE_GUARD_NAME)
+ set(_include_guard "ECM_GENERATEEXPORTHEADER_${ARGS_INCLUDE_GUARD_NAME}")
+ else()
+ set(_include_guard "ECM_GENERATEEXPORTHEADER_${_upper_base_name}_EXPORT_H")
+ endif()
+
+ file(APPEND ${_header_work_file} "
+
+#ifndef ${_include_guard}
+#define ${_include_guard}
+
+${_output}
+
+#endif // ${_include_guard}
+"
+ )
+ endif()
+
+ # avoid rebuilding if there was no change
+ execute_process(
+ COMMAND ${CMAKE_COMMAND} -E copy_if_different "${_header_work_file}" "${_header_file}"
+ )
+ file(REMOVE "${_header_work_file}")
+endfunction()