Browse code

Add project

Robert Cranston authored on 17/12/2021 12:48:41
Showing 4 changed files

1 1
new file mode 100644
... ...
@@ -0,0 +1 @@
1
+/build
0 2
new file mode 100644
... ...
@@ -0,0 +1,18 @@
1
+## CMake
2
+cmake_minimum_required(VERSION 3.14)
3
+if(NOT CMAKE_BUILD_TYPE)
4
+    set(CMAKE_BUILD_TYPE Debug)
5
+endif()
6
+
7
+## Project
8
+project(glquad
9
+    VERSION 1.0.0
10
+    LANGUAGES
11
+)
12
+
13
+## Main target
14
+add_custom_target(${PROJECT_NAME} ALL)
15
+
16
+## C++
17
+include(cxx.cmake)
18
+cxx()
... ...
@@ -8,6 +8,94 @@ A [GLSL][]/[OpenGL][] [\>=3.2][] [quad][] library.
8 8
 [\>=3.2]: https://en.wikipedia.org/wiki/OpenGL#Version_history
9 9
 [quad]: https://en.wikipedia.org/wiki/Quadrilateral
10 10
 
11
+## Build system
12
+
13
+This project has support for [CMake][], and uses [`cmake-cxx`][].
14
+
15
+There are several ways to use this project in another CMake-based project:
16
+
17
+-   Use [`FetchContent`][] to download it and import the targets automatically
18
+    as part of the configure step:
19
+
20
+    ```cmake
21
+    include(FetchContent)
22
+    FetchContent_Declare(project-name
23
+        GIT_REPOSITORY https://example.com/user/project-name
24
+    )
25
+    FetchContent_MakeAvailable(
26
+        project-name
27
+    )
28
+    ```
29
+
30
+-   Bundle it and import the targets with [`add_subdirectory`][].
31
+
32
+-   Use [`find_package`][] (requires that the project is [packaged](#packaging)
33
+    or [installed](#installing)).
34
+
35
+As usual, use [`add_dependencies`][]/[`target_link_libraries`][] to declare the
36
+dependency.
37
+
38
+[CMake]: https://cmake.org
39
+[`cmake-cxx`]: https://git.rcrnstn.net/rcrnstn/cmake-cxx
40
+[`FetchContent`]: https://cmake.org/cmake/help/v3.14/module/FetchContent.html
41
+[`add_subdirectory`]: https://cmake.org/cmake/help/v3.14/command/add_subdirectory.html
42
+[`find_package`]: https://cmake.org/cmake/help/v3.14/command/find_package.html
43
+[`add_dependencies`]: https://cmake.org/cmake/help/v3.14/command/add_dependencies.html
44
+[`target_link_libraries`]: https://cmake.org/cmake/help/v3.14/command/target_link_libraries.html
45
+
46
+### Configure and generate
47
+
48
+To configure and generate a build tree, use `cmake`:
49
+
50
+```sh
51
+cmake -B build
52
+```
53
+
54
+To set the [`CMAKE_BUILD_TYPE`][], pass e.g. `-DCMAKE_BUILD_TYPE=Release`.
55
+
56
+[`cmake`]: https://cmake.org/cmake/help/v3.14/manual/cmake.1.html#generate-a-project-buildsystem
57
+[`CMAKE_BUILD_TYPE`]: https://cmake.org/cmake/help/v3.14/variable/CMAKE_BUILD_TYPE.html
58
+
59
+### Build
60
+
61
+To build, use [`cmake --build`][]:
62
+
63
+```sh
64
+cmake --build build
65
+```
66
+
67
+[`cmake --build`]: https://cmake.org/cmake/help/v3.14/manual/cmake.1.html#build-a-project
68
+
69
+### Test
70
+
71
+To run tests, use [`ctest`][]:
72
+
73
+```sh
74
+(cd build && ctest --output-on-failure)
75
+```
76
+
77
+[`ctest`]: https://cmake.org/cmake/help/v3.14/manual/ctest.1.html
78
+
79
+### Package
80
+
81
+To package, use [`cpack`][]:
82
+
83
+```sh
84
+(cd build && cpack)
85
+```
86
+
87
+[`cpack`]: https://cmake.org/cmake/help/v3.14/manual/cpack.1.html
88
+
89
+### Install
90
+
91
+To install onto the current system, use [`cmake --install`][]:
92
+
93
+```sh
94
+cmake --install build
95
+```
96
+
97
+[`cmake --install`]: https://cmake.org/cmake/help/v3.14/manual/cmake.1.html#install-a-project
98
+
11 99
 ## License
12 100
 
13 101
 Licensed under the [ISC License][] unless otherwise noted, see the
14 102
new file mode 100644
... ...
@@ -0,0 +1,310 @@
1
+# https://git.rcrnstn.net/rcrnstn/cmake-cxx
2
+
3
+function(cxx)
4
+    ## Arguments
5
+    set(OPTIONS
6
+        DISABLE_SANITIZERS
7
+        DISABLE_CPPCHECK
8
+        DISABLE_CLANG_TIDY
9
+        DISABLE_INCLUDE_WHAT_YOU_USE
10
+    )
11
+    set(ONE_VALUE_ARGS
12
+        CXX_STANDARD
13
+    )
14
+    set(MULTI_VALUE_ARGS
15
+        PACKAGES
16
+        FETCHCONTENT
17
+        DEPENDENCIES_PUBLIC
18
+        DEPENDENCIES_PRIVATE
19
+        DEPENDENCIES_TESTS
20
+        DEFINES
21
+    )
22
+    cmake_parse_arguments(PARSE_ARGV 0 ARG
23
+        "${OPTIONS}"
24
+        "${ONE_VALUE_ARGS}"
25
+        "${MULTI_VALUE_ARGS}"
26
+    )
27
+
28
+    ## Helper variables
29
+    set(NON_INTERFACE_TARGETS)
30
+    set(PROJECT_IS_INTERFACE_LIBRARY)
31
+    set(PROJECT_IS_TOP_LEVEL)
32
+    get_target_property(PROJECT_TYPE ${PROJECT_NAME} TYPE)
33
+    if("${PROJECT_TYPE}" STREQUAL "INTERFACE_LIBRARY")
34
+        set(PROJECT_IS_INTERFACE_LIBRARY TRUE)
35
+    endif()
36
+    if("${CMAKE_PROJECT_NAME}" STREQUAL "${PROJECT_NAME}")
37
+        set(PROJECT_IS_TOP_LEVEL TRUE)
38
+    endif()
39
+
40
+    ## CMake modules
41
+    include(FetchContent)
42
+    include(CMakePackageConfigHelpers)
43
+    include(CheckIPOSupported)
44
+    include(GNUInstallDirs)
45
+    if(PROJECT_IS_TOP_LEVEL)
46
+        include(CTest)
47
+        include(CPack)
48
+    endif()
49
+
50
+    ## Packages
51
+    foreach(PACKAGE ${ARG_PACKAGES})
52
+        find_package(${PACKAGE} REQUIRED)
53
+    endforeach()
54
+
55
+    ## Fetch content
56
+    foreach(FETCHCONTENT ${ARG_FETCHCONTENT})
57
+        get_filename_component(FETCHCONTENT_NAME ${FETCHCONTENT} NAME)
58
+        FetchContent_Declare(${FETCHCONTENT_NAME}
59
+            GIT_REPOSITORY ${FETCHCONTENT}
60
+        )
61
+        FetchContent_MakeAvailable(${FETCHCONTENT_NAME})
62
+    endforeach()
63
+
64
+    ## Configure main target
65
+    file(GLOB_RECURSE SRC CONFIGURE_DEPENDS
66
+        src/*
67
+    )
68
+    file(GLOB_RECURSE INCLUDE CONFIGURE_DEPENDS
69
+        include/*
70
+    )
71
+    if(NOT ("${SRC}" STREQUAL "" AND "${INCLUDE}" STREQUAL ""))
72
+        target_sources(${PROJECT_NAME}
73
+            PRIVATE
74
+                ${SRC}
75
+                ${INCLUDE}
76
+        )
77
+        set_property(TARGET ${PROJECT_NAME} APPEND PROPERTY
78
+            PUBLIC_HEADER ${INCLUDE}
79
+        )
80
+        if(PROJECT_IS_INTERFACE_LIBRARY)
81
+            target_include_directories(${PROJECT_NAME}
82
+                INTERFACE
83
+                    "$<BUILD_INTERFACE:${PROJECT_SOURCE_DIR}/include>"
84
+                    "$<INSTALL_INTERFACE:include>"
85
+            )
86
+        else()
87
+            target_include_directories(${PROJECT_NAME}
88
+                PUBLIC
89
+                    "$<BUILD_INTERFACE:${PROJECT_SOURCE_DIR}/include>"
90
+                    "$<INSTALL_INTERFACE:include>"
91
+                PRIVATE
92
+                    src
93
+            )
94
+            list(APPEND NON_INTERFACE_TARGETS ${PROJECT_NAME})
95
+        endif()
96
+        target_link_libraries(${PROJECT_NAME}
97
+            PUBLIC
98
+                ${ARG_DEPENDENCIES_PUBLIC}
99
+            PRIVATE
100
+                "$<BUILD_INTERFACE:${ARG_DEPENDENCIES_PRIVATE}>"
101
+        )
102
+        if(PROJECT_IS_TOP_LEVEL)
103
+            install(TARGETS ${PROJECT_NAME} EXPORT ${PROJECT_NAME})
104
+            install(EXPORT ${PROJECT_NAME}
105
+                FILE ${PROJECT_NAME}Config.cmake
106
+                DESTINATION ${CMAKE_INSTALL_LIBDIR}/cmake/${PROJECT_NAME}
107
+            )
108
+            write_basic_package_version_file(${PROJECT_NAME}ConfigVersion.cmake
109
+                COMPATIBILITY SameMajorVersion
110
+            )
111
+            install(
112
+                FILES ${PROJECT_BINARY_DIR}/${PROJECT_NAME}ConfigVersion.cmake
113
+                DESTINATION ${CMAKE_INSTALL_LIBDIR}/cmake/${PROJECT_NAME}
114
+            )
115
+        endif()
116
+    endif()
117
+
118
+    ## Declare and configure test targets
119
+    if(PROJECT_IS_TOP_LEVEL AND BUILD_TESTING)
120
+        file(GLOB_RECURSE TESTS_COMMON CONFIGURE_DEPENDS
121
+            tests/common/*
122
+        )
123
+        file(GLOB TESTS LIST_DIRECTORIES FALSE CONFIGURE_DEPENDS
124
+            tests/*
125
+        )
126
+        foreach(TEST ${TESTS})
127
+            get_filename_component(TEST_NAME "${TEST}" NAME_WE)
128
+            set(TEST_TARGET "${PROJECT_NAME}-test-${TEST_NAME}")
129
+            add_executable(${TEST_TARGET}
130
+                ${TEST}
131
+                ${TESTS_COMMON}
132
+            )
133
+            target_include_directories(${TEST_TARGET}
134
+                PRIVATE
135
+                    tests
136
+            )
137
+            target_link_libraries(${TEST_TARGET}
138
+                PRIVATE
139
+                    ${PROJECT_NAME}
140
+                    ${ARG_DEPENDENCIES_TESTS}
141
+            )
142
+            add_test(
143
+                NAME    ${TEST_TARGET}
144
+                COMMAND ${TEST_TARGET}
145
+            )
146
+            list(APPEND NON_INTERFACE_TARGETS ${TEST_TARGET})
147
+        endforeach()
148
+    endif()
149
+
150
+    ## Declare and configure assets
151
+    file(REMOVE_RECURSE
152
+        ${CMAKE_BINARY_DIR}/assets
153
+    )
154
+    file(GLOB_RECURSE ASSETS RELATIVE "${PROJECT_SOURCE_DIR}" CONFIGURE_DEPENDS
155
+        assets/*
156
+    )
157
+    if(NOT "${ASSETS}" STREQUAL "")
158
+        add_custom_target(${PROJECT_NAME}-assets)
159
+        add_dependencies(${PROJECT_NAME} ${PROJECT_NAME}-assets)
160
+    endif()
161
+    foreach(ASSET ${ASSETS})
162
+        get_filename_component(ASSET_DIR ${ASSET} DIRECTORY)
163
+        add_custom_command(
164
+            DEPENDS
165
+                ${PROJECT_SOURCE_DIR}/${ASSET}
166
+            OUTPUT
167
+                ${CMAKE_BINARY_DIR}/${ASSET}
168
+            COMMAND
169
+                ${CMAKE_COMMAND} -E make_directory
170
+                    ${CMAKE_BINARY_DIR}/${ASSET_DIR}
171
+            COMMAND
172
+                ${CMAKE_COMMAND} -E create_symlink
173
+                    ${PROJECT_SOURCE_DIR}/${ASSET}
174
+                    ${CMAKE_BINARY_DIR}/${ASSET}
175
+            VERBATIM
176
+        )
177
+        set_property(TARGET ${PROJECT_NAME}-assets APPEND PROPERTY
178
+            SOURCES ${CMAKE_BINARY_DIR}/${ASSET}
179
+        )
180
+        if(NOT "${ASSET}" MATCHES "/tests/")
181
+            install(
182
+                FILES
183
+                    ${PROJECT_SOURCE_DIR}/${ASSET}
184
+                DESTINATION
185
+                    ${CMAKE_INSTALL_DATADIR}/${CMAKE_PROJECT_NAME}/${ASSET_DIR}
186
+            )
187
+        endif()
188
+    endforeach()
189
+
190
+    ## Preprocessor defines
191
+    set_property(TARGET ${NON_INTERFACE_TARGETS} APPEND PROPERTY
192
+        COMPILE_DEFINITIONS ${ARG_DEFINES}
193
+    )
194
+
195
+    ## Language version
196
+    if(NOT "${CXX_STANDARD}" STREQUAL "")
197
+        set_target_properties(${NON_INTERFACE_TARGETS} PROPERTIES
198
+            CXX_STANDARD          ${ARG_CXX_STANDARD}
199
+            CXX_STANDARD_REQUIRED ON
200
+            CXX_EXTENSIONS        OFF
201
+        )
202
+    endif()
203
+
204
+    ## Build options
205
+    if(MSVC)
206
+        set(COMPILE_OPTIONS
207
+            /permissive-
208
+            /WX
209
+            /W4
210
+        )
211
+    elseif("${CMAKE_CXX_COMPILER_ID}" MATCHES "GNU|.*Clang")
212
+        if(NOT ARG_DISABLE_SANITIZERS)
213
+            set(SANITIZE_OPTIONS
214
+                $<$<CONFIG:Debug>:
215
+                    -g -fno-omit-frame-pointer
216
+                    -fsanitize=address,leak,undefined
217
+                    -fsanitize=float-divide-by-zero
218
+                    -fsanitize=float-cast-overflow
219
+                    # -fsanitize=pointer-compare
220
+                    # -fsanitize=pointer-subtract
221
+                    # -fsanitize=implicit-conversion
222
+                    -fno-sanitize-recover=all
223
+                >
224
+            )
225
+        endif()
226
+        set(COMPILE_OPTIONS
227
+            -pedantic
228
+            -Werror # -Wfatal-errors
229
+            -Wall -Wextra
230
+            -Wconversion -Wsign-conversion
231
+            -Wdouble-promotion
232
+            -Wimplicit-fallthrough
233
+            -Wvla
234
+            # -Wzero-as-null-pointer-constant
235
+            # -Wshadow
236
+            -Weffc++
237
+            ${SANITIZE_OPTIONS}
238
+        )
239
+        set(LINK_OPTIONS
240
+            ${SANITIZE_OPTIONS}
241
+        )
242
+        check_ipo_supported(RESULT INTERPROCEDURAL_OPTIMIZATION)
243
+    endif()
244
+    set_property(TARGET ${NON_INTERFACE_TARGETS} APPEND PROPERTY
245
+        COMPILE_OPTIONS "${COMPILE_OPTIONS}"
246
+    )
247
+    set_property(TARGET ${NON_INTERFACE_TARGETS} APPEND PROPERTY
248
+        LINK_OPTIONS "${LINK_OPTIONS}"
249
+    )
250
+    set_property(TARGET ${NON_INTERFACE_TARGETS} APPEND PROPERTY
251
+        INTERPROCEDURAL_OPTIMIZATION "${INTERPROCEDURAL_OPTIMIZATION}"
252
+    )
253
+
254
+    ## Sanitizer environment variables
255
+    set(SAN_STRIP_PATH_PREFIX strip_path_prefix=${PROJECT_BINARY_DIR})
256
+    set(ASAN_OPTIONS  "${SAN_STRIP_PATH_PREFIX}")
257
+    set(LSAN_OPTIONS  "${SAN_STRIP_PATH_PREFIX}")
258
+    set(UBSAN_OPTIONS "${SAN_STRIP_PATH_PREFIX}")
259
+    file(GLOB ASAN_SUPP CONFIGURE_DEPENDS
260
+        .asan.supp
261
+    )
262
+    file(GLOB LSAN_SUPP CONFIGURE_DEPENDS
263
+        .lsan.supp
264
+    )
265
+    file(GLOB UBSAN_SUPP CONFIGURE_DEPENDS
266
+        .ubsan.supp
267
+    )
268
+    if(NOT "${ASAN_SUPP}" STREQUAL "")
269
+        set(ASAN_OPTIONS "${ASAN_OPTIONS},suppressions=${ASAN_SUPP}")
270
+    endif()
271
+    if(NOT "${LSAN_SUPP}" STREQUAL "")
272
+        set(LSAN_OPTIONS "${LSAN_OPTIONS},suppressions=${LSAN_SUPP}")
273
+    endif()
274
+    if(NOT "${UBSAN_SUPP}" STREQUAL "")
275
+        set(UBSAN_OPTIONS "${UBSAN_OPTIONS},suppressions=${UBSAN_SUPP}")
276
+    endif()
277
+    # set(ASAN_OPTIONS "${ASAN_OPTIONS},detect_leaks=1")
278
+    # set(UBSAN_OPTIONS "${UBSAN_OPTIONS},print_stacktrace=1")
279
+    set_property(TEST ${NON_INTERFACE_TARGETS} APPEND PROPERTY
280
+        ENVIRONMENT
281
+            ASAN_OPTIONS=${ASAN_OPTIONS}
282
+            LSAN_OPTIONS=${LSAN_OPTIONS}
283
+            UBSAN_OPTIONS=${UBSAN_OPTIONS}
284
+    )
285
+
286
+    ## Tools
287
+    if(PROJECT_IS_TOP_LEVEL)
288
+        find_program(CPPCHECK             cppcheck)
289
+        find_program(CLANG_TIDY           clang-tidy)
290
+        find_program(INCLUDE_WHAT_YOU_USE include-what-you-use)
291
+        if(CPPCHECK AND NOT ARG_DISABLE_CPPCHECK)
292
+            set(CXX_CPPCHECK ${CPPCHECK}
293
+                --enable=warning,style
294
+                --error-exitcode=1
295
+            )
296
+        endif()
297
+        if(CLANG_TIDY AND NOT ARG_DISABLE_CLANG_TIDY)
298
+            set(CXX_CLANG_TIDY ${CLANG_TIDY})
299
+        endif()
300
+        if(INCLUDE_WHAT_YOU_USE AND NOT ARG_DISABLE_INCLUDE_WHAT_YOU_USE)
301
+            set(CXX_INCLUDE_WHAT_YOU_USE ${INCLUDE_WHAT_YOU_USE})
302
+        endif()
303
+        set_target_properties(${NON_INTERFACE_TARGETS} PROPERTIES
304
+            EXPORT_COMPILE_COMMANDS  ON
305
+            CXX_CPPCHECK             "${CXX_CPPCHECK}"
306
+            CXX_CLANG_TIDY           "${CXX_CLANG_TIDY}"
307
+            CXX_INCLUDE_WHAT_YOU_USE "${CXX_INCLUDE_WHAT_YOU_USE}"
308
+        )
309
+    endif()
310
+endfunction()