-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathCMakeLists.txt
210 lines (168 loc) · 9.86 KB
/
CMakeLists.txt
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
# =============================================================================
# Copyright (c) 2024, NVIDIA CORPORATION.
#
# Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except
# in compliance with the License. You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software distributed under the License
# is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express
# or implied. See the License for the specific language governing permissions and limitations under
# the License.
# =============================================================================
# cmake-lint: disable=R0915
include_guard(GLOBAL)
cmake_minimum_required(VERSION 3.26.4 FATAL_ERROR)
include(rapids_config.cmake)
include(rapids-cmake)
include(rapids-cpm)
project(
RAPIDS_LOGGER
# TODO: Determine appropriate versioning.
VERSION 0.0.1
LANGUAGES CXX)
rapids_cmake_build_type(Release)
rapids_cpm_init()
#[=======================================================================[.rst:
rapids_make_logger
------------------
Generate a logger implementation customized for the specified namespace.
.. code-block:: cmake
rapids_make_logger(<logger_namespace>
[EXPORT_SET <export-set-name>]
[LOGGER_TARGET <logger-target-name>]
[LOGGER_HEADER_DIR <header-dir>]
[LOGGER_MACRO_PREFIX <macro-prefix>]
[LOGGER_DEFAULT_LEVEL <logger-default-level>]
[CMAKE_ALIAS_NAMESPACE <alias-namespace>]
)
This function produces an interface target named <logger-target-name> that, when linked to by other targets, provides them a header file that defines a logger interface for the specified namespace. The logger is generated from the provided template files and is configured to use the specified namespace and macro prefix. The generated logger is placed in the specified header directory.
The logger implementation lives in a separate header file that is not included in the declaration header. The logger implementation is compiled into a separate target that must be linked to by any target that uses the logger.
``logger_namespace``
The namespace for which to generate the logger implementation.
``EXPORT_SET``
The name of the export set to which the logger target should be added. If not specified, the logger target is not added to any export set.
``LOGGER_TARGET``
The name of the logger (and logger impl) target to create. If not specified, <logger-target-name> defaults to <logger_namespace>_logger.
``LOGGER_HEADER_DIR``
The directory in which to place the generated logger header file. If not specified, the logger header file is placed in include/<logger_namespace>.
``LOGGER_MACRO_PREFIX``
The prefix to use for the logger macros. If not specified, the macro prefix is the uppercase version of the logger namespace.
``LOGGER_DEFAULT_LEVEL``
The default level used by the logger at runtime. If not specified, defaults to "INFO".
``CMAKE_ALIAS_NAMESPACE``
The namespace to use for the CMake alias target. If not specified, the alias namespace is the same as the logger namespace.
Result Targets
^^^^^^^^^^^^^^^^
<logger-target-name> is an interface target that provides the logger interface for the specified namespace.
<logger-target-name>_impl is an interface target that provides the logger implementation for the specified namespace. This target must be linked to by any target that uses the logger. Targets linking to this target will have the logger implementation compiled into them.
Examples
^^^^^^^^
Example on how to use :cmake:command:`rapids_make_logger`.
.. code-block:: cmake
# Generate a logger for the namespace "rapids" with default level WARN and
# associate it with the export set "rapids-exports".
rapids_make_logger(rapids
EXPORT_SET rapids-exports
LOGGER_DEFAULT_LEVEL WARN
)
#]=======================================================================]
function(rapids_make_logger logger_namespace)
list(APPEND CMAKE_MESSAGE_CONTEXT "rapids_make_logger")
set(_rapids_options)
set(_rapids_one_value EXPORT_SET LOGGER_TARGET LOGGER_HEADER_DIR LOGGER_MACRO_PREFIX LOGGER_DEFAULT_LEVEL)
set(_rapids_multi_value)
cmake_parse_arguments(_RAPIDS "${_rapids_options}" "${_rapids_one_value}"
"${_rapids_multi_value}" ${ARGN})
# Most arguments are optional and can be inferred from the namespace by default.
set(_RAPIDS_LOGGER_NAMESPACE ${logger_namespace})
if(NOT _RAPIDS_LOGGER_TARGET)
set(_RAPIDS_LOGGER_TARGET "${logger_namespace}_logger")
endif()
if(NOT _RAPIDS_LOGGER_HEADER_DIR)
set(_RAPIDS_LOGGER_HEADER_DIR "include/${logger_namespace}")
endif()
if(NOT _RAPIDS_LOGGER_MACRO_PREFIX)
string(TOUPPER ${logger_namespace} _RAPIDS_LOGGER_MACRO_PREFIX)
endif()
if(NOT _RAPIDS_LOGGER_DEFAULT_LEVEL)
set(_RAPIDS_LOGGER_DEFAULT_LEVEL "INFO")
endif()
if(NOT _RAPIDS_CMAKE_NAMESPACE_ALIAS)
set(_RAPIDS_CMAKE_NAMESPACE_ALIAS ${logger_namespace})
endif()
# Get spdlog
include(${rapids-cmake-dir}/cpm/spdlog.cmake)
rapids_cpm_spdlog(
# The conda package for fmt is hard-coded to assume that we use a preexisting fmt library. This
# is why we have always had a libfmt linkage despite choosing to specify the header-only version
# of fmt. We need a more robust way of modifying this to support fully self-contained build and
# usage even in environments where fmt and/or spdlog are already present. The crudest solution
# would be to modify the interface compile definitions and link libraries of the spdlog target,
# if necessary. For now I'm specifying EXTERNAL_FMT_HO here so that in environments where spdlog
# is cloned and built from source we wind up with the behavior that we expect, but we'll have to
# resolve this properly eventually.
FMT_OPTION "EXTERNAL_FMT_HO"
# TODO: For consistency with the rest of our CMake we would want these to be separately
# configurable, but in the ideal end state we probably want to wind up with a BYO spdlog setup
# so that you can build against even header-only libraries using rapids-logger without spdlog if
# you don't want to use the logging functionality (e.g. you don't need spdlog for building
# against rmm if aren't using the logger targets) so I'll leave this fixed here for now in
# anticipation of a long-term fix that removes these altogether.
INSTALL_EXPORT_SET ${_RAPIDS_EXPORT_SET}
BUILD_EXPORT_SET ${_RAPIDS_EXPORT_SET})
# All paths are computed relative to the current source/binary dir of the file from which the
# function is invoked. As a result we cannot use relative paths here because CMake will root these
# paths incorrectly for configure_file/install.
set(BUILD_DIR ${CMAKE_CURRENT_BINARY_DIR}/${_RAPIDS_LOGGER_HEADER_DIR})
# TODO: Verify that installation works correctly with prefix removed.
set(INSTALL_DIR ${_RAPIDS_LOGGER_HEADER_DIR})
set(LOGGER_OUTPUT_FILE ${BUILD_DIR}/logger.hpp)
configure_file(${CMAKE_CURRENT_FUNCTION_LIST_DIR}/logger.hpp.in ${LOGGER_OUTPUT_FILE})
install(FILES ${LOGGER_OUTPUT_FILE} DESTINATION ${INSTALL_DIR})
set(LOGGER_IMPL_OUTPUT_FILE ${BUILD_DIR}/logger_impl/logger_impl.hpp)
configure_file(${CMAKE_CURRENT_FUNCTION_LIST_DIR}/logger_impl.hpp.in ${LOGGER_IMPL_OUTPUT_FILE})
install(FILES ${LOGGER_IMPL_OUTPUT_FILE} DESTINATION ${INSTALL_DIR}/logger_impl)
add_library(${_RAPIDS_LOGGER_TARGET} INTERFACE)
add_library(${_RAPIDS_CMAKE_NAMESPACE_ALIAS}::${_RAPIDS_LOGGER_TARGET} ALIAS ${_RAPIDS_LOGGER_TARGET})
include(GNUInstallDirs)
# Note: The BUILD_INTERFACE setting assumes that LOGGER_HEADER_DIR is the subdirectory of
# CMAKE_INSTALL_INCLUDEDIR relative to which all includes are rooted in the C++ code files. I
# think that is a safe assumption though since if it were violated then the INSTALL_INTERFACE
# would not only be incorrect (if computed using LOGGER_HEADER_DIR), but it would also break
# consumers of the installed package who expect to be able to write `#include
# <${LOGGER_HEADER_DIR/include\//}/logger.hpp>` and have it work.
target_include_directories(
${_RAPIDS_LOGGER_TARGET}
INTERFACE "$<BUILD_INTERFACE:${CMAKE_CURRENT_BINARY_DIR}/${CMAKE_INSTALL_INCLUDEDIR}>"
"$<INSTALL_INTERFACE:${CMAKE_INSTALL_INCLUDEDIR}>")
target_compile_features(${_RAPIDS_LOGGER_TARGET} INTERFACE cxx_std_17)
# Create an interface target that will trigger compilation of the logger implementation in any
# target that is linked to it.
set(LOGGER_IMPL_SRC_OUTPUT_FILE ${BUILD_DIR}/logger_impl/logger.cpp)
configure_file(${CMAKE_CURRENT_FUNCTION_LIST_DIR}/logger.cpp.in ${LOGGER_IMPL_SRC_OUTPUT_FILE})
install(FILES ${LOGGER_IMPL_SRC_OUTPUT_FILE} DESTINATION ${INSTALL_DIR}/logger_impl)
# Note that we cannot specify the source files directly in add_library, see the CMake
# documentation explaining that these do not populate INTERFACE_SOURCES.
# https://cmake.org/cmake/help/latest/command/add_library.html#interface-with-sources
set(impl_target ${_RAPIDS_LOGGER_TARGET}_impl)
add_library(${impl_target} INTERFACE)
add_library(${_RAPIDS_CMAKE_NAMESPACE_ALIAS}::${impl_target} ALIAS ${impl_target})
target_sources(
${impl_target}
INTERFACE $<BUILD_INTERFACE:${LOGGER_IMPL_SRC_OUTPUT_FILE}>
$<INSTALL_INTERFACE:${_RAPIDS_LOGGER_HEADER_DIR}/logger_impl/logger.cpp>)
target_link_libraries(${impl_target} INTERFACE ${_RAPIDS_LOGGER_TARGET}
spdlog::spdlog_header_only)
set_target_properties(${impl_target} PROPERTIES POSITION_INDEPENDENT_CODE ON
INTERFACE_POSITION_INDEPENDENT_CODE ON)
set(_install_export)
if(_RAPIDS_EXPORT_SET)
set(_install_export EXPORT ${_RAPIDS_EXPORT_SET})
endif()
install(TARGETS ${_RAPIDS_LOGGER_TARGET} ${_install_export})
if(TARGET ${impl_target})
install(TARGETS ${impl_target} ${_install_export})
endif()
endfunction()