onqtam | 4a65563 | 2016-05-26 14:20:52 +0300 | [diff] [blame] | 1 | if(common_included) |
| 2 | return() |
| 3 | endif() |
| 4 | set(common_included true) |
| 5 | |
| 6 | include(CMakeParseArguments) |
| 7 | |
| 8 | # cache this for use inside of the function |
| 9 | set(CURRENT_LIST_DIR_CACHED ${CMAKE_CURRENT_LIST_DIR}) |
| 10 | |
| 11 | enable_testing() |
| 12 | |
| 13 | set(TEST_MODE "COMPARE" CACHE STRING "Test mode - normal/run through valgrind/collect output/compare with output") |
| 14 | set_property(CACHE TEST_MODE PROPERTY STRINGS "NORMAL;VALGRIND;COLLECT;COMPARE") |
onqtam | 5cab8e9 | 2016-10-10 17:58:38 +0300 | [diff] [blame] | 15 | option(WITH_CPP11 "With C++11 enabled" OFF) |
onqtam | 4a65563 | 2016-05-26 14:20:52 +0300 | [diff] [blame] | 16 | |
onqtam | db5eee9 | 2016-09-15 17:12:50 +0300 | [diff] [blame] | 17 | # a custom version of add_test() to suite my needs |
| 18 | function(doctest_add_test) |
onqtam | 4a65563 | 2016-05-26 14:20:52 +0300 | [diff] [blame] | 19 | cmake_parse_arguments(ARG "NO_VALGRIND;NO_OUTPUT" "NAME" "COMMAND" ${ARGN}) |
| 20 | if(NOT "${ARG_UNPARSED_ARGUMENTS}" STREQUAL "" OR "${ARG_NAME}" STREQUAL "" OR "${ARG_COMMAND}" STREQUAL "") |
onqtam | db5eee9 | 2016-09-15 17:12:50 +0300 | [diff] [blame] | 21 | message(FATAL_ERROR "doctest_add_test() called with wrong options!") |
onqtam | 4a65563 | 2016-05-26 14:20:52 +0300 | [diff] [blame] | 22 | endif() |
| 23 | |
| 24 | set(the_test_mode NORMAL) |
| 25 | |
| 26 | # construct the command that will be called by the exec_test.cmake script |
| 27 | set(the_command "") |
| 28 | if(${TEST_MODE} STREQUAL "VALGRIND" AND NOT ARG_NO_VALGRIND) |
| 29 | set(the_test_mode VALGRIND) |
| 30 | set(the_command "valgrind -v --leak-check=full --track-origins=yes --error-exitcode=1") |
| 31 | endif() |
| 32 | foreach(cur ${ARG_COMMAND}) |
| 33 | set(the_command "${the_command} ${cur}") |
| 34 | endforeach() |
| 35 | # append the argument for removing paths from filenames in the output so tests give the same output everywhere |
| 36 | set(the_command "${the_command} --dt-no-path-filenames=1") |
onqtam | 1fc3dc7 | 2017-03-14 14:30:09 +0200 | [diff] [blame] | 37 | # append the argument for substituting source line numbers with 0 in the output so tests give the same output when lines change a bit |
| 38 | set(the_command "${the_command} --dt-no-line-numbers=1") |
onqtam | 9f934f8 | 2016-08-02 12:42:19 +0300 | [diff] [blame] | 39 | # append the argument for ignoring the exit code of the test programs because some are intended to have failing tests |
onqtam | 4a65563 | 2016-05-26 14:20:52 +0300 | [diff] [blame] | 40 | set(the_command "${the_command} --dt-no-exitcode=1") |
onqtam | 0b51b7b | 2016-11-08 14:27:30 +0200 | [diff] [blame] | 41 | # append the argument for not printing the framework version so reference output doesn't have to be recommitted when the version is bumped |
| 42 | set(the_command "${the_command} --dt-no-version=1") |
onqtam | 4a65563 | 2016-05-26 14:20:52 +0300 | [diff] [blame] | 43 | |
| 44 | string(STRIP ${the_command} the_command) |
| 45 | |
| 46 | if(${TEST_MODE} STREQUAL "COLLECT" OR ${TEST_MODE} STREQUAL "COMPARE") |
| 47 | if(NOT ARG_NO_OUTPUT) |
| 48 | file(MAKE_DIRECTORY ${CMAKE_CURRENT_SOURCE_DIR}/test_output/) |
| 49 | set(the_test_mode ${TEST_MODE}) |
| 50 | list(APPEND ADDITIONAL_FLAGS -DTEST_OUTPUT_FILE=${CMAKE_CURRENT_SOURCE_DIR}/test_output/${ARG_NAME}.txt) |
| 51 | list(APPEND ADDITIONAL_FLAGS -DTEST_TEMP_FILE=${CMAKE_CURRENT_BINARY_DIR}${CMAKE_FILES_DIRECTORY}/temp_test_output.txt) |
| 52 | endif() |
| 53 | endif() |
| 54 | |
| 55 | list(APPEND ADDITIONAL_FLAGS -DTEST_MODE=${the_test_mode}) |
| 56 | |
onqtam | db5eee9 | 2016-09-15 17:12:50 +0300 | [diff] [blame] | 57 | add_test(NAME ${ARG_NAME} COMMAND ${CMAKE_COMMAND} -DCOMMAND=${the_command} ${ADDITIONAL_FLAGS} -P ${CURRENT_LIST_DIR_CACHED}/exec_test.cmake) |
onqtam | 4a65563 | 2016-05-26 14:20:52 +0300 | [diff] [blame] | 58 | endfunction() |
| 59 | |
| 60 | macro(add_compiler_flags) |
| 61 | foreach(flag ${ARGV}) |
| 62 | set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} ${flag}") |
| 63 | endforeach() |
| 64 | endmacro() |
| 65 | |
| 66 | if(CMAKE_CXX_COMPILER_ID MATCHES "GNU|Clang") |
| 67 | add_compiler_flags(-Werror) |
onqtam | 4a65563 | 2016-05-26 14:20:52 +0300 | [diff] [blame] | 68 | add_compiler_flags(-pedantic) |
| 69 | add_compiler_flags(-pedantic-errors) |
| 70 | add_compiler_flags(-fvisibility=hidden) |
| 71 | add_compiler_flags(-fstrict-aliasing) |
onqtam | 5cab8e9 | 2016-10-10 17:58:38 +0300 | [diff] [blame] | 72 | |
| 73 | if(WITH_CPP11) |
| 74 | add_compiler_flags(-std=c++0x) |
| 75 | else() |
| 76 | add_compiler_flags(-std=c++98) |
| 77 | endif() |
onqtam | 4a65563 | 2016-05-26 14:20:52 +0300 | [diff] [blame] | 78 | endif() |
| 79 | |
| 80 | if(CMAKE_CXX_COMPILER_ID MATCHES "GNU") |
onqtam | 4a65563 | 2016-05-26 14:20:52 +0300 | [diff] [blame] | 81 | add_compiler_flags(-Wall) |
| 82 | add_compiler_flags(-Wextra) |
onqtam | 4a65563 | 2016-05-26 14:20:52 +0300 | [diff] [blame] | 83 | add_compiler_flags(-fdiagnostics-show-option) |
| 84 | add_compiler_flags(-Wconversion) |
onqtam | 4a65563 | 2016-05-26 14:20:52 +0300 | [diff] [blame] | 85 | add_compiler_flags(-Wold-style-cast) |
| 86 | add_compiler_flags(-Wfloat-equal) |
| 87 | add_compiler_flags(-Wlogical-op) |
| 88 | add_compiler_flags(-Wundef) |
| 89 | add_compiler_flags(-Wredundant-decls) |
| 90 | add_compiler_flags(-Wshadow) |
| 91 | add_compiler_flags(-Wstrict-overflow=5) |
| 92 | add_compiler_flags(-Wwrite-strings) |
| 93 | add_compiler_flags(-Wpointer-arith) |
| 94 | add_compiler_flags(-Wcast-qual) |
| 95 | add_compiler_flags(-Wformat=2) |
| 96 | add_compiler_flags(-Wswitch-default) |
| 97 | add_compiler_flags(-Wmissing-include-dirs) |
| 98 | add_compiler_flags(-Wcast-align) |
onqtam | 4a65563 | 2016-05-26 14:20:52 +0300 | [diff] [blame] | 99 | add_compiler_flags(-Wswitch-enum) |
onqtam | 4a65563 | 2016-05-26 14:20:52 +0300 | [diff] [blame] | 100 | add_compiler_flags(-Wnon-virtual-dtor) |
onqtam | 4a65563 | 2016-05-26 14:20:52 +0300 | [diff] [blame] | 101 | add_compiler_flags(-Wctor-dtor-privacy) |
onqtam | 4a65563 | 2016-05-26 14:20:52 +0300 | [diff] [blame] | 102 | add_compiler_flags(-Wsign-conversion) |
| 103 | add_compiler_flags(-Wdisabled-optimization) |
| 104 | add_compiler_flags(-Weffc++) |
| 105 | add_compiler_flags(-Winline) |
| 106 | add_compiler_flags(-Winvalid-pch) |
onqtam | 4a65563 | 2016-05-26 14:20:52 +0300 | [diff] [blame] | 107 | add_compiler_flags(-Wmissing-declarations) |
| 108 | add_compiler_flags(-Woverloaded-virtual) |
onqtam | 4a65563 | 2016-05-26 14:20:52 +0300 | [diff] [blame] | 109 | if(NOT CMAKE_CXX_COMPILER_VERSION VERSION_LESS 4.6) |
| 110 | add_compiler_flags(-Wnoexcept) |
| 111 | endif() |
| 112 | #add_compiler_flags(-Waggregate-return) # GCC 4.8 does not silence this even with "#pragma GCC diagnostic ignored" |
| 113 | |
| 114 | if(NOT CMAKE_CXX_COMPILER_VERSION VERSION_LESS 5.0) |
| 115 | add_compiler_flags(-Wdouble-promotion) |
| 116 | add_compiler_flags(-Wtrampolines) |
| 117 | add_compiler_flags(-Wzero-as-null-pointer-constant) |
| 118 | add_compiler_flags(-Wuseless-cast) |
| 119 | add_compiler_flags(-Wvector-operation-performance) |
| 120 | add_compiler_flags(-Wsized-deallocation) |
| 121 | endif() |
| 122 | |
onqtam | b382bd3 | 2016-08-01 20:53:56 +0300 | [diff] [blame] | 123 | if(CMAKE_CXX_COMPILER_VERSION VERSION_EQUAL 6.0 OR CMAKE_CXX_COMPILER_VERSION VERSION_GREATER 6.0) |
onqtam | 4a65563 | 2016-05-26 14:20:52 +0300 | [diff] [blame] | 124 | add_compiler_flags(-Wshift-overflow=2) |
| 125 | add_compiler_flags(-Wnull-dereference) |
| 126 | add_compiler_flags(-Wduplicated-cond) |
onqtam | 4a65563 | 2016-05-26 14:20:52 +0300 | [diff] [blame] | 127 | endif() |
onqtam | f63c510 | 2017-02-25 20:00:52 +0200 | [diff] [blame] | 128 | |
| 129 | if(CMAKE_CXX_COMPILER_VERSION VERSION_EQUAL 7.0 OR CMAKE_CXX_COMPILER_VERSION VERSION_GREATER 7.0) |
| 130 | add_compiler_flags(-Walloc-zero) |
| 131 | add_compiler_flags(-Walloca) |
| 132 | endif() |
onqtam | 4a65563 | 2016-05-26 14:20:52 +0300 | [diff] [blame] | 133 | endif() |
| 134 | |
| 135 | if(CMAKE_CXX_COMPILER_ID MATCHES "Clang") |
| 136 | add_compiler_flags(-Weverything) |
| 137 | add_compiler_flags(-Qunused-arguments -fcolor-diagnostics) # needed for ccache integration on travis |
| 138 | endif() |
| 139 | |
| 140 | if(CMAKE_CXX_COMPILER_ID MATCHES "MSVC") |
onqtam | 8edeaaf | 2016-08-16 07:12:58 +0300 | [diff] [blame] | 141 | add_compiler_flags(/std:c++latest) # for post c++14 updates in MSVC |
onqtam | 4a65563 | 2016-05-26 14:20:52 +0300 | [diff] [blame] | 142 | add_compiler_flags(/WX) |
| 143 | add_compiler_flags(/W4) # /Wall is too aggressive - even the standard C headers give thousands of errors... |
| 144 | endif() |
onqtam | 7ffa84e | 2016-05-30 18:42:20 +0300 | [diff] [blame] | 145 | |
| 146 | # add a custom target that assembles the single header when any of the parts are touched |
| 147 | |
| 148 | set(cat_cmd "cat") |
| 149 | set(doctest_include_folder "${CURRENT_LIST_DIR_CACHED}/../doctest/") |
| 150 | set(doctest_parts_folder "${CURRENT_LIST_DIR_CACHED}/../doctest/parts/") |
| 151 | if(WIN32) |
| 152 | set(cat_cmd "type") |
| 153 | STRING(REGEX REPLACE "/" "\\\\" doctest_include_folder ${doctest_include_folder}) |
| 154 | STRING(REGEX REPLACE "/" "\\\\" doctest_parts_folder ${doctest_parts_folder}) |
| 155 | endif() |
| 156 | |
| 157 | add_custom_command( |
| 158 | OUTPUT ${doctest_include_folder}doctest.h |
| 159 | DEPENDS |
| 160 | ${doctest_parts_folder}doctest_fwd.h |
| 161 | ${doctest_parts_folder}doctest_impl.h |
onqtam | f720d43 | 2016-05-31 17:51:10 +0300 | [diff] [blame] | 162 | COMMAND ${CMAKE_COMMAND} -P ${CURRENT_LIST_DIR_CACHED}/asemble_single_header.cmake |
onqtam | 7ffa84e | 2016-05-30 18:42:20 +0300 | [diff] [blame] | 163 | COMMENT "assembling the single header") |
| 164 | |
| 165 | add_custom_target(assemble_single_header ALL DEPENDS ${doctest_include_folder}doctest.h) |
onqtam | 29e6260 | 2016-05-30 19:45:10 +0300 | [diff] [blame] | 166 | |
| 167 | # override add_executable() to add a dependency on the header assembly target |
| 168 | function(add_executable name) |
| 169 | _add_executable(${name} ${ARGN}) |
| 170 | add_dependencies(${name} assemble_single_header) |
| 171 | endfunction() |
| 172 | |
| 173 | # override add_library() to add a dependency on the header assembly target |
| 174 | function(add_library name) |
| 175 | _add_library(${name} ${ARGN}) |
| 176 | add_dependencies(${name} assemble_single_header) |
| 177 | endfunction() |