xref: /aosp_15_r20/external/boringssl/src/CMakeLists.txt (revision 8fb009dc861624b67b6cdb62ea21f0f22d0c584b)
1cmake_minimum_required(VERSION 3.12)
2
3# Defer enabling C and CXX languages.
4project(BoringSSL NONE)
5
6# Don't install BoringSSL to system directories by default; it has no stable
7# ABI. Instead, default to an "install" directory under the source.
8if(CMAKE_INSTALL_PREFIX_INITIALIZED_TO_DEFAULT)
9  set(CMAKE_INSTALL_PREFIX ${CMAKE_SOURCE_DIR}/install CACHE PATH "" FORCE)
10endif()
11
12if(WIN32)
13  # On Windows, prefer cl over gcc if both are available. By default most of
14  # the CMake generators prefer gcc, even on Windows.
15  set(CMAKE_GENERATOR_CC cl)
16endif()
17
18include(cmake/go.cmake)
19include(cmake/paths.cmake)
20include(gen/sources.cmake)
21
22enable_language(C)
23enable_language(CXX)
24
25include(GNUInstallDirs)
26
27set(INSTALL_ENABLED 1)
28
29if(CMAKE_SYSTEM_NAME STREQUAL "Linux" AND NOT CMAKE_CROSSCOMPILING)
30  find_package(PkgConfig QUIET)
31  if (PkgConfig_FOUND)
32    pkg_check_modules(LIBUNWIND libunwind-generic>=1.3.0)
33    if(LIBUNWIND_FOUND)
34      add_definitions(-DBORINGSSL_HAVE_LIBUNWIND)
35    else()
36      message("libunwind not found. Disabling unwind tests.")
37    endif()
38  else()
39    message("pkgconfig not found. Disabling unwind tests.")
40  endif()
41endif()
42
43string(TOLOWER "${CMAKE_BUILD_TYPE}" CMAKE_BUILD_TYPE_LOWER)
44if(NOT FIPS)
45  if(CMAKE_BUILD_TYPE_LOWER STREQUAL "relwithassert" OR
46     NOT CMAKE_BUILD_TYPE_LOWER MATCHES "rel")
47    add_definitions(-DBORINGSSL_DISPATCH_TEST)
48    # CMake automatically connects include_directories to the NASM
49    # command-line, but not add_definitions.
50    set(CMAKE_ASM_NASM_FLAGS "${CMAKE_ASM_NASM_FLAGS} -DBORINGSSL_DISPATCH_TEST")
51  endif()
52endif()
53
54# Add a RelWithAsserts build configuration. It is the same as Release, except it
55# does not define NDEBUG, so asserts run.
56foreach(VAR CMAKE_C_FLAGS CMAKE_CXX_FLAGS CMAKE_ASM_FLAGS)
57  string(REGEX REPLACE "(^| )[/-]DNDEBUG( |$)" " " "${VAR}_RELWITHASSERTS"
58         "${${VAR}_RELEASE}")
59endforeach()
60
61if(BORINGSSL_PREFIX AND BORINGSSL_PREFIX_SYMBOLS)
62  require_go()
63  add_definitions(-DBORINGSSL_PREFIX=${BORINGSSL_PREFIX})
64  # CMake automatically connects include_directories to the NASM command-line,
65  # but not add_definitions.
66  set(CMAKE_ASM_NASM_FLAGS "${CMAKE_ASM_NASM_FLAGS} -DBORINGSSL_PREFIX=${BORINGSSL_PREFIX}")
67
68  # Use "symbol_prefix_include" to store generated header files
69  include_directories(${CMAKE_CURRENT_BINARY_DIR}/symbol_prefix_include)
70  add_custom_command(
71    OUTPUT symbol_prefix_include/boringssl_prefix_symbols.h
72           symbol_prefix_include/boringssl_prefix_symbols_asm.h
73           symbol_prefix_include/boringssl_prefix_symbols_nasm.inc
74    COMMAND ${CMAKE_COMMAND} -E make_directory ${CMAKE_CURRENT_BINARY_DIR}/symbol_prefix_include
75    COMMAND ${GO_EXECUTABLE} run ${CMAKE_CURRENT_SOURCE_DIR}/util/make_prefix_headers.go -out ${CMAKE_CURRENT_BINARY_DIR}/symbol_prefix_include ${BORINGSSL_PREFIX_SYMBOLS}
76    DEPENDS util/make_prefix_headers.go
77            ${BORINGSSL_PREFIX_SYMBOLS})
78
79  # add_dependencies needs a target, not a file, so we add an intermediate
80  # target.
81  add_custom_target(
82    boringssl_prefix_symbols
83    DEPENDS symbol_prefix_include/boringssl_prefix_symbols.h
84            symbol_prefix_include/boringssl_prefix_symbols_asm.h
85            symbol_prefix_include/boringssl_prefix_symbols_nasm.inc)
86elseif(BORINGSSL_PREFIX OR BORINGSSL_PREFIX_SYMBOLS)
87  message(FATAL_ERROR "Must specify both or neither of BORINGSSL_PREFIX and BORINGSSL_PREFIX_SYMBOLS")
88else()
89  add_custom_target(boringssl_prefix_symbols)
90endif()
91
92if(CMAKE_CXX_COMPILER_ID MATCHES "Clang")
93  set(CLANG 1)
94endif()
95
96if(CMAKE_SYSTEM_NAME STREQUAL "Emscripten")
97  set(EMSCRIPTEN 1)
98endif()
99
100set(CMAKE_CXX_STANDARD 14)
101set(CMAKE_CXX_STANDARD_REQUIRED ON)
102set(CMAKE_C_STANDARD 11)
103set(CMAKE_C_STANDARD_REQUIRED ON)
104
105if(CMAKE_COMPILER_IS_GNUCXX OR CLANG)
106  # Note clang-cl is odd and sets both CLANG and MSVC. We base our configuration
107  # primarily on our normal Clang one.
108  set(C_CXX_FLAGS "-Werror -Wformat=2 -Wsign-compare -Wwrite-strings -Wvla -Wshadow -Wtype-limits -Wmissing-field-initializers")
109  if(MSVC)
110    # clang-cl sets different default warnings than clang. It also treats -Wall
111    # as -Weverything, to match MSVC. Instead -W3 is the alias for -Wall.
112    # See http://llvm.org/viewvc/llvm-project?view=revision&revision=319116
113    set(C_CXX_FLAGS "${C_CXX_FLAGS} -W3 -Wno-unused-parameter -fmsc-version=1900")
114  else()
115    if(EMSCRIPTEN)
116      # emscripten's emcc/clang does not accept the "-ggdb" flag.
117      set(C_CXX_FLAGS "${C_CXX_FLAGS} -g")
118    else()
119      set(C_CXX_FLAGS "${C_CXX_FLAGS} -ggdb")
120    endif()
121
122    set(C_CXX_FLAGS "${C_CXX_FLAGS} -Wall -fvisibility=hidden -fno-common")
123  endif()
124
125  if(CLANG)
126    set(C_CXX_FLAGS "${C_CXX_FLAGS} -Wnewline-eof -fcolor-diagnostics")
127  else()
128    # GCC (at least 4.8.4) has a bug where it'll find unreachable free() calls
129    # and declare that the code is trying to free a stack pointer.
130    set(C_CXX_FLAGS "${C_CXX_FLAGS} -Wno-free-nonheap-object")
131  endif()
132
133  # -Wstring-concatenation was added in Clang 12.0.0, which corresponds to
134  # AppleClang 13.0.0 per the table in
135  # https://en.wikipedia.org/wiki/Xcode#Toolchain_versions
136  if((CMAKE_C_COMPILER_ID STREQUAL "Clang" AND
137      CMAKE_C_COMPILER_VERSION VERSION_GREATER_EQUAL "12.0.0") OR
138     (CMAKE_C_COMPILER_ID STREQUAL "AppleClang" AND
139      CMAKE_C_COMPILER_VERSION VERSION_GREATER_EQUAL "13.0.0"))
140    set(C_CXX_FLAGS "${C_CXX_FLAGS} -Wstring-concatenation")
141  endif()
142
143  # Clang 12's -Wframe-larger-than reportedly does not work in clang-cl. See
144  # https://crbug.com/boringssl/709. Clang 13 includes the following fix, which
145  # may be related. Speculatively gate on Clang 13. That corresponds to
146  # AppleClang 13.1.6.
147  # https://github.com/llvm/llvm-project/commit/6aaf4fa2885600b0e31042071ad06f78218ab0f2
148  if((CMAKE_C_COMPILER_ID STREQUAL "Clang" AND
149      CMAKE_C_COMPILER_VERSION VERSION_GREATER_EQUAL "13.0.0") OR
150     (CMAKE_C_COMPILER_ID STREQUAL "AppleClang" AND
151      CMAKE_C_COMPILER_VERSION VERSION_GREATER_EQUAL "13.1.6"))
152    set(C_CXX_FLAGS "${C_CXX_FLAGS} -Wframe-larger-than=25344")
153  endif()
154
155  # -Wctad-maybe-unsupported was added in Clang 10, which is AppleClang 12.0.0.
156  if((CMAKE_C_COMPILER_ID STREQUAL "Clang" AND
157      CMAKE_C_COMPILER_VERSION VERSION_GREATER_EQUAL "10.0.0") OR
158     (CMAKE_C_COMPILER_ID STREQUAL "AppleClang" AND
159      CMAKE_C_COMPILER_VERSION VERSION_GREATER_EQUAL "12.0.0"))
160    set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -Wctad-maybe-unsupported")
161  endif()
162
163  if(CLANG OR CMAKE_C_COMPILER_VERSION VERSION_GREATER_EQUAL "7.0.0")
164    set(C_CXX_FLAGS "${C_CXX_FLAGS} -Wimplicit-fallthrough")
165  endif()
166
167  if(CMAKE_COMPILER_IS_GNUCXX)
168    set(C_CXX_FLAGS "${C_CXX_FLAGS} -Wformat-signedness")
169  endif()
170
171  set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} ${C_CXX_FLAGS} -Wmissing-prototypes -Wold-style-definition -Wstrict-prototypes")
172  set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} ${C_CXX_FLAGS} -Wmissing-declarations")
173
174  # In GCC, -Wmissing-declarations is the C++ spelling of -Wmissing-prototypes
175  # and using the wrong one is an error. In Clang, -Wmissing-prototypes is the
176  # spelling for both and -Wmissing-declarations is some other warning.
177  #
178  # https://gcc.gnu.org/onlinedocs/gcc-7.1.0/gcc/Warning-Options.html#Warning-Options
179  # https://clang.llvm.org/docs/DiagnosticsReference.html#wmissing-prototypes
180  # https://clang.llvm.org/docs/DiagnosticsReference.html#wmissing-declarations
181  if(CLANG)
182    set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -Wmissing-prototypes")
183  endif()
184elseif(MSVC)
185  set(MSVC_DISABLED_WARNINGS_LIST
186      "C4100" # 'exarg' : unreferenced formal parameter
187      "C4127" # conditional expression is constant
188      "C4244" # 'function' : conversion from 'int' to 'uint8_t',
189              # possible loss of data
190      "C4267" # conversion from 'size_t' to 'int', possible loss of data
191      "C4706" # assignment within conditional expression
192      )
193  string(REPLACE "C" " -wd" MSVC_DISABLED_WARNINGS_STR
194                            ${MSVC_DISABLED_WARNINGS_LIST})
195  set(CMAKE_C_FLAGS   "-utf-8 -W4 -WX ${MSVC_DISABLED_WARNINGS_STR}")
196  # Without /Zc:__cplusplus, MSVC does not define the right value for
197  # __cplusplus. See https://devblogs.microsoft.com/cppblog/msvc-now-correctly-reports-__cplusplus/
198  # If this becomes too problematic for downstream code, we can look at
199  # _MSVC_LANG.
200  set(CMAKE_CXX_FLAGS "-utf-8 -W4 -WX ${MSVC_DISABLED_WARNINGS_STR} -Zc:__cplusplus")
201endif()
202
203if(WIN32)
204  add_definitions(-D_HAS_EXCEPTIONS=0)
205  add_definitions(-DWIN32_LEAN_AND_MEAN)
206  add_definitions(-DNOMINMAX)
207  # Allow use of fopen.
208  add_definitions(-D_CRT_SECURE_NO_WARNINGS)
209endif()
210
211# pthread_rwlock_t on Linux requires a feature flag. We limit this to Linux
212# because, on Apple platforms, it instead disables APIs we use. See compat(5)
213# and sys/cdefs.h. Reportedly, FreeBSD also breaks when this is set. See
214# https://crbug.com/boringssl/471.
215if(CMAKE_SYSTEM_NAME STREQUAL "Linux")
216  set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -D_XOPEN_SOURCE=700")
217endif()
218
219if(FUZZ)
220  if(NOT CLANG)
221    message(FATAL_ERROR "You need to build with Clang for fuzzing to work")
222  endif()
223
224  if(CMAKE_C_COMPILER_VERSION VERSION_LESS "6.0.0")
225    message(FATAL_ERROR "You need Clang ≥ 6.0.0")
226  endif()
227
228  add_definitions(-DBORINGSSL_UNSAFE_DETERMINISTIC_MODE)
229  set(RUNNER_ARGS "-deterministic")
230
231  if(NOT NO_FUZZER_MODE)
232    add_definitions(-DBORINGSSL_UNSAFE_FUZZER_MODE)
233    set(RUNNER_ARGS ${RUNNER_ARGS} "-fuzzer" "-shim-config" "fuzzer_mode.json")
234  endif()
235
236  set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -fsanitize=address,fuzzer-no-link -fsanitize-coverage=edge,indirect-calls")
237  set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -fsanitize=address,fuzzer-no-link -fsanitize-coverage=edge,indirect-calls")
238endif()
239
240add_definitions(-DBORINGSSL_IMPLEMENTATION)
241
242if(BUILD_SHARED_LIBS)
243  add_definitions(-DBORINGSSL_SHARED_LIBRARY)
244  # Enable position-independent code globally. This is needed because
245  # some library targets are OBJECT libraries.
246  set(CMAKE_POSITION_INDEPENDENT_CODE TRUE)
247endif()
248
249if(MSAN)
250  if(NOT CLANG)
251    message(FATAL_ERROR "Cannot enable MSAN unless using Clang")
252  endif()
253
254  if(ASAN)
255    message(FATAL_ERROR "ASAN and MSAN are mutually exclusive")
256  endif()
257
258  set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -fsanitize=memory -fsanitize-memory-track-origins -fno-omit-frame-pointer")
259  set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -fsanitize=memory -fsanitize-memory-track-origins -fno-omit-frame-pointer")
260  set(CMAKE_ASM_FLAGS "${CMAKE_ASM_FLAGS} -fsanitize=memory -fsanitize-memory-track-origins -fno-omit-frame-pointer")
261endif()
262
263if(ASAN)
264  if(NOT CLANG)
265    message(FATAL_ERROR "Cannot enable ASAN unless using Clang")
266  endif()
267
268  set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -fsanitize=address -fsanitize-address-use-after-scope -fno-omit-frame-pointer")
269  set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -fsanitize=address -fsanitize-address-use-after-scope -fno-omit-frame-pointer")
270endif()
271
272if(CFI)
273  if(NOT CLANG)
274    message(FATAL_ERROR "Cannot enable CFI unless using Clang")
275  endif()
276
277  set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -fsanitize=cfi -fno-sanitize-trap=cfi -flto=thin")
278  set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -fsanitize=cfi -fno-sanitize-trap=cfi -flto=thin")
279  # We use Chromium's copy of clang, which requires -fuse-ld=lld if building
280  # with -flto. That, in turn, can't handle -ggdb.
281  set(CMAKE_EXE_LINKER_FLAGS "${CMAKE_EXE_LINKER_FLAGS} -fuse-ld=lld")
282  string(REPLACE "-ggdb" "-g" CMAKE_C_FLAGS "${CMAKE_C_FLAGS}")
283  string(REPLACE "-ggdb" "-g" CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS}")
284  # -flto causes object files to contain LLVM bitcode. Mixing those with
285  # assembly output in the same static library breaks the linker.
286  set(OPENSSL_NO_ASM "1")
287endif()
288
289if(TSAN)
290  if(NOT CLANG)
291    message(FATAL_ERROR "Cannot enable TSAN unless using Clang")
292  endif()
293
294  set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -fsanitize=thread")
295  set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -fsanitize=thread")
296  set(CMAKE_EXE_LINKER_FLAGS "${CMAKE_EXE_LINKER_FLAGS} -fsanitize=thread")
297endif()
298
299if(UBSAN)
300  if(NOT CLANG)
301    message(FATAL_ERROR "Cannot enable UBSAN unless using Clang")
302  endif()
303
304  set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -fsanitize=undefined")
305  set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -fsanitize=undefined")
306  set(CMAKE_EXE_LINKER_FLAGS "${CMAKE_EXE_LINKER_FLAGS} -fsanitize=undefined")
307
308  if(NOT UBSAN_RECOVER)
309    set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -fno-sanitize-recover=undefined")
310    set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -fno-sanitize-recover=undefined")
311    set(CMAKE_EXE_LINKER_FLAGS "${CMAKE_EXE_LINKER_FLAGS} -fno-sanitize-recover=undefined")
312  endif()
313endif()
314
315if(GCOV)
316  set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -fprofile-arcs -ftest-coverage")
317  set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -fprofile-arcs -ftest-coverage")
318endif()
319
320if(FIPS)
321  require_go()
322  add_definitions(-DBORINGSSL_FIPS)
323  if(FIPS_BREAK_TEST)
324    add_definitions("-DBORINGSSL_FIPS_BREAK_${FIPS_BREAK_TEST}=1")
325  endif()
326  # The FIPS integrity check does not work for ASan and MSan builds.
327  if(NOT ASAN AND NOT MSAN)
328    if(BUILD_SHARED_LIBS)
329      set(FIPS_SHARED "1")
330    else()
331      set(FIPS_DELOCATE "1")
332    endif()
333  endif()
334  if(FIPS_SHARED)
335    # The Android CMake files set -ffunction-sections and -fdata-sections,
336    # which is incompatible with FIPS_SHARED.
337    set(CMAKE_C_FLAGS
338        "${CMAKE_C_FLAGS} -fno-function-sections -fno-data-sections")
339    set(CMAKE_CXX_FLAGS
340        "${CMAKE_CXX_FLAGS} -fno-function-sections -fno-data-sections")
341  endif()
342endif()
343
344if(OPENSSL_SMALL)
345  add_definitions(-DOPENSSL_SMALL)
346endif()
347
348if(CONSTANT_TIME_VALIDATION)
349  add_definitions(-DBORINGSSL_CONSTANT_TIME_VALIDATION)
350endif()
351
352if(MALLOC_FAILURE_TESTING)
353  add_definitions(-DBORINGSSL_MALLOC_FAILURE_TESTING)
354endif()
355
356if(OPENSSL_NO_ASM)
357  add_definitions(-DOPENSSL_NO_ASM)
358endif()
359
360if(FIPS_DELOCATE OR NOT OPENSSL_NO_ASM)
361  # On x86 and x86_64 Windows, we use the NASM output.
362  if(WIN32 AND CMAKE_SYSTEM_PROCESSOR MATCHES "AMD64|x86_64|amd64|x86|i[3-6]86")
363    enable_language(ASM_NASM)
364    set(OPENSSL_NASM TRUE)
365    set(CMAKE_ASM_NASM_FLAGS "${CMAKE_ASM_NASM_FLAGS} -gcv8")
366  else()
367    enable_language(ASM)
368    set(OPENSSL_ASM TRUE)
369    # Work around https://gitlab.kitware.com/cmake/cmake/-/issues/20771 in older
370    # CMake versions.
371    if(APPLE AND CMAKE_VERSION VERSION_LESS 3.19)
372      if(CMAKE_OSX_SYSROOT)
373        set(CMAKE_ASM_FLAGS "${CMAKE_ASM_FLAGS} -isysroot \"${CMAKE_OSX_SYSROOT}\"")
374      endif()
375      foreach(arch ${CMAKE_OSX_ARCHITECTURES})
376        set(CMAKE_ASM_FLAGS "${CMAKE_ASM_FLAGS} -arch ${arch}")
377      endforeach()
378    endif()
379    if(NOT WIN32)
380      set(CMAKE_ASM_FLAGS "${CMAKE_ASM_FLAGS} -Wa,--noexecstack")
381    endif()
382    # Clang's integerated assembler does not support debug symbols.
383    if(NOT CMAKE_ASM_COMPILER_ID MATCHES "Clang")
384      set(CMAKE_ASM_FLAGS "${CMAKE_ASM_FLAGS} -Wa,-g")
385    endif()
386  endif()
387endif()
388
389if(OPENSSL_NO_SSE2_FOR_TESTING)
390  add_definitions(-DOPENSSL_NO_SSE2_FOR_TESTING)
391endif()
392
393if(USE_CUSTOM_LIBCXX)
394  if(NOT CLANG)
395    message(FATAL_ERROR "USE_CUSTOM_LIBCXX only supported with Clang")
396  endif()
397
398  # CMake does not allow installing a library without installing dependencies.
399  # If we installed libcrypto, we'd have to install our custom libc++, which
400  # does not make sense. As this is a test-only configuration, disable
401  # installing.
402  set(INSTALL_ENABLED 0)
403
404  # CMAKE_CXX_FLAGS ends up in the linker flags as well, so use
405  # add_compile_options. There does not appear to be a way to set
406  # language-specific compile-only flags.
407  add_compile_options("-nostdinc++")
408  set(CMAKE_CXX_LINK_FLAGS "${CMAKE_CXX_LINK_FLAGS} -nostdlib++")
409  include_directories(
410    SYSTEM
411    util/bot/libcxx-config
412    util/bot/libcxx/include
413    util/bot/libcxxabi/include
414  )
415
416  # This is patterned after buildtools/third_party/libc++/BUILD.gn and
417  # buildtools/third_party/libc++abi/BUILD.gn in Chromium.
418
419  file(GLOB LIBCXX_SOURCES "util/bot/libcxx/src/*.cpp")
420  file(GLOB LIBCXXABI_SOURCES "util/bot/libcxxabi/src/*.cpp")
421
422  # This file is meant for exception-less builds.
423  list(REMOVE_ITEM LIBCXXABI_SOURCES "trunk/src/cxa_noexception.cpp")
424  # libc++ also defines new and delete.
425  list(REMOVE_ITEM LIBCXXABI_SOURCES "trunk/src/stdlib_new_delete.cpp")
426  if(TSAN)
427    # ThreadSanitizer tries to intercept these symbols. Skip them to avoid
428    # symbol conflicts.
429    list(REMOVE_ITEM LIBCXXABI_SOURCES "trunk/src/cxa_guard.cpp")
430  endif()
431
432  add_library(libcxxabi ${LIBCXXABI_SOURCES})
433  target_compile_definitions(
434    libcxxabi PRIVATE
435    -D_LIBCPP_ENABLE_CXX17_REMOVED_UNEXPECTED_FUNCTIONS
436  )
437
438  add_library(libcxx ${LIBCXX_SOURCES})
439  if(ASAN OR MSAN OR TSAN)
440    # Sanitizers try to intercept new and delete.
441    target_compile_definitions(
442      libcxx PRIVATE
443      -D_LIBCPP_DISABLE_NEW_DELETE_DEFINITIONS
444    )
445  endif()
446  target_compile_definitions(
447    libcxx PRIVATE
448    -D_LIBCPP_BUILDING_LIBRARY
449    -DLIBCXX_BUILDING_LIBCXXABI
450  )
451  set_target_properties(
452    libcxx libcxxabi PROPERTIES
453    COMPILE_FLAGS "-Wno-missing-prototypes -Wno-implicit-fallthrough"
454    # libc++ and libc++abi must be built in C++23 mode.
455    CXX_STANDARD 23
456    CXX_STANDARD_REQUIRED TRUE
457  )
458  # libc++abi depends on libc++ internal headers.
459  set_property(TARGET libcxx libcxxabi APPEND PROPERTY INCLUDE_DIRECTORIES "${CMAKE_CURRENT_SOURCE_DIR}/util/bot/libcxx/src")
460  target_link_libraries(libcxx libcxxabi)
461endif()
462
463# Add minimal googletest targets. The provided one has many side-effects, and
464# googletest has a very straightforward build.
465add_library(
466  boringssl_gtest
467  third_party/googletest/googlemock/src/gmock-all.cc
468  third_party/googletest/googletest/src/gtest-all.cc
469)
470if(USE_CUSTOM_LIBCXX)
471  target_link_libraries(boringssl_gtest libcxx)
472endif()
473target_include_directories(
474    boringssl_gtest
475    PUBLIC
476    third_party/googletest/googlemock/include
477    third_party/googletest/googletest/include
478    PRIVATE
479    third_party/googletest/googlemock
480    third_party/googletest/googletest
481)
482
483# Declare a dummy target to build all unit tests. Test targets should inject
484# themselves as dependencies next to the target definition.
485add_custom_target(all_tests)
486
487add_subdirectory(ssl/test)
488add_subdirectory(util/fipstools)
489add_subdirectory(util/fipstools/acvp/modulewrapper)
490
491if(OPENSSL_ASM)
492  set(CRYPTO_SOURCES_ASM_USED ${CRYPTO_SOURCES_ASM})
493  set(BCM_SOURCES_ASM_USED ${BCM_SOURCES_ASM})
494  set(TEST_SUPPORT_SOURCES_ASM_USED ${TEST_SUPPORT_SOURCES_ASM})
495elseif(OPENSSL_NASM)
496  set(CRYPTO_SOURCES_ASM_USED ${CRYPTO_SOURCES_NASM})
497  set(BCM_SOURCES_ASM_USED ${BCM_SOURCES_NASM})
498  set(TEST_SUPPORT_SOURCES_ASM_USED ${TEST_SUPPORT_SOURCES_NASM})
499endif()
500
501if(FIPS_DELOCATE AND FIPS_SHARED)
502  message(FATAL_ERROR "Can't set both delocate and shared mode for FIPS build")
503endif()
504
505if(FIPS_DELOCATE)
506  add_library(bcm_c_generated_asm STATIC ${BCM_SOURCES})
507  add_dependencies(bcm_c_generated_asm boringssl_prefix_symbols)
508  target_include_directories(bcm_c_generated_asm PRIVATE ${PROJECT_SOURCE_DIR}/include)
509  set_target_properties(bcm_c_generated_asm PROPERTIES COMPILE_OPTIONS "-S")
510  set_target_properties(bcm_c_generated_asm PROPERTIES POSITION_INDEPENDENT_CODE ON)
511
512  set(TARGET_FLAG "")
513  if(CMAKE_ASM_COMPILER_TARGET)
514    set(TARGET_FLAG "--target=${CMAKE_ASM_COMPILER_TARGET}")
515  endif()
516
517  go_executable(delocate boringssl.googlesource.com/boringssl/util/fipstools/delocate)
518  add_custom_command(
519    OUTPUT bcm-delocated.S
520    COMMAND ${CMAKE_CURRENT_BINARY_DIR}/delocate
521            -a $<TARGET_FILE:bcm_c_generated_asm>
522            -o ${CMAKE_CURRENT_BINARY_DIR}/bcm-delocated.S
523            -cc ${CMAKE_ASM_COMPILER}
524            -cc-flags "${TARGET_FLAG} ${CMAKE_ASM_FLAGS}"
525            ${BCM_SOURCES_ASM_USED}
526            ${CRYPTO_HEADERS}
527    DEPENDS bcm_c_generated_asm
528            delocate
529            ${BCM_SOURCES_ASM_USED}
530            ${CRYPTO_HEADERS}
531    WORKING_DIRECTORY ${CMAKE_CURRENT_SOURCE_DIR}
532  )
533
534  add_library(bcm_hashunset STATIC bcm-delocated.S)
535  set_target_properties(bcm_hashunset PROPERTIES POSITION_INDEPENDENT_CODE ON)
536  set_target_properties(bcm_hashunset PROPERTIES LINKER_LANGUAGE C)
537
538  go_executable(inject_hash
539                boringssl.googlesource.com/boringssl/util/fipstools/inject_hash)
540  add_custom_command(
541    OUTPUT bcm.o
542    COMMAND ./inject_hash -o bcm.o -in-archive $<TARGET_FILE:bcm_hashunset>
543    DEPENDS bcm_hashunset inject_hash
544    WORKING_DIRECTORY ${CMAKE_CURRENT_BINARY_DIR}
545  )
546  set(CRYPTO_FIPS_OBJECTS bcm.o)
547elseif(FIPS_SHARED)
548  if(NOT BUILD_SHARED_LIBS)
549    message(FATAL_ERROR "FIPS_SHARED set but not BUILD_SHARED_LIBS")
550  endif()
551
552  add_library(bcm_library STATIC ${BCM_SOURCES} ${BCM_SOURCES_ASM_USED})
553  add_dependencies(bcm_library boringssl_prefix_symbols)
554  target_include_directories(bcm_library PRIVATE ${PROJECT_SOURCE_DIR}/include)
555
556  add_custom_command(
557    OUTPUT bcm.o
558    COMMAND ${CMAKE_LINKER} -r -T ${CMAKE_CURRENT_SOURCE_DIR}/crypto/fipsmodule/fips_shared.lds -o bcm.o --whole-archive $<TARGET_FILE:bcm_library>
559    DEPENDS bcm_library crypto/fipsmodule/fips_shared.lds
560    WORKING_DIRECTORY ${CMAKE_CURRENT_BINARY_DIR}
561  )
562  set(CRYPTO_FIPS_OBJECTS bcm.o)
563else()
564  add_library(fipsmodule OBJECT ${BCM_SOURCES} ${BCM_SOURCES_ASM_USED})
565  add_dependencies(fipsmodule boringssl_prefix_symbols)
566  target_include_directories(fipsmodule PRIVATE ${PROJECT_SOURCE_DIR}/include)
567  set(CRYPTO_FIPS_OBJECTS $<TARGET_OBJECTS:fipsmodule>)
568endif()
569
570add_library(crypto ${CRYPTO_SOURCES} ${CRYPTO_FIPS_OBJECTS} ${CRYPTO_SOURCES_ASM_USED})
571target_include_directories(crypto PUBLIC
572  $<BUILD_INTERFACE:${PROJECT_SOURCE_DIR}/include>
573  $<INSTALL_INTERFACE:include>
574)
575set_property(TARGET crypto PROPERTY EXPORT_NAME Crypto)
576
577if(FIPS_SHARED)
578  # Rewrite libcrypto.so to inject the correct module hash value. This assumes
579  # UNIX-style library naming, but we only support FIPS mode on Linux anyway.
580  add_custom_command(
581    TARGET crypto POST_BUILD
582    COMMAND ${GO_EXECUTABLE} run
583    ${CMAKE_CURRENT_SOURCE_DIR}/util/fipstools/inject_hash/inject_hash.go
584    -o libcrypto.so -in-object libcrypto.so
585    # The DEPENDS argument to a POST_BUILD rule appears to be ignored. Thus
586    # go_executable isn't used (as it doesn't get built), but we list this
587    # dependency anyway in case it starts working in some CMake version.
588    DEPENDS util/fipstools/inject_hash/inject_hash.go
589    WORKING_DIRECTORY ${CMAKE_CURRENT_BINARY_DIR}
590  )
591endif()
592
593add_dependencies(crypto boringssl_prefix_symbols)
594if(WIN32)
595  target_link_libraries(crypto ws2_32)
596endif()
597
598# CMAKE_SYSTEM_NAME is "Generic" for embedded OSes:
599# https://cmake.org/cmake/help/book/mastering-cmake/chapter/Cross%20Compiling%20With%20CMake.html#toolchain-files
600#
601# For now we assume embedded OSes do not have threads. Additionally, the Threads
602# package does not work with Android, but Android does not require any extra
603# parameters to link pthreads.
604if(NOT CMAKE_SYSTEM_NAME MATCHES "^(Generic|Android)$")
605  find_package(Threads REQUIRED)
606  target_link_libraries(crypto Threads::Threads)
607endif()
608
609# Every target depends on crypto, so we add libcxx as a dependency here to
610# simplify injecting it everywhere.
611if(USE_CUSTOM_LIBCXX)
612  target_link_libraries(crypto libcxx)
613endif()
614
615add_library(ssl ${SSL_SOURCES})
616# Although libssl also provides headers that require an include directory, the
617# flag is already specified by libcrypto, so we omit target_include_directories
618# here.
619set_property(TARGET ssl PROPERTY EXPORT_NAME SSL)
620target_link_libraries(ssl crypto)
621
622add_library(decrepit ${DECREPIT_SOURCES})
623target_link_libraries(decrepit crypto ssl)
624
625add_library(test_support_lib STATIC
626            ${TEST_SUPPORT_SOURCES} ${TEST_SUPPORT_SOURCES_ASM_USED})
627if(LIBUNWIND_FOUND)
628  target_compile_options(test_support_lib PRIVATE ${LIBUNWIND_CFLAGS_OTHER})
629  target_include_directories(test_support_lib PRIVATE ${LIBUNWIND_INCLUDE_DIRS})
630  target_link_libraries(test_support_lib ${LIBUNWIND_LDFLAGS})
631endif()
632if(WIN32)
633  target_link_libraries(test_support_lib dbghelp)
634endif()
635target_link_libraries(test_support_lib boringssl_gtest crypto)
636
637# urandom_test is a separate binary because it needs to be able to observe the
638# PRNG initialisation, which means that it can't have other tests running before
639# it does.
640add_executable(urandom_test ${URANDOM_TEST_SOURCES})
641target_link_libraries(urandom_test test_support_lib boringssl_gtest crypto)
642add_dependencies(all_tests urandom_test)
643
644add_executable(crypto_test ${CRYPTO_TEST_SOURCES})
645target_link_libraries(crypto_test test_support_lib boringssl_gtest crypto)
646add_dependencies(all_tests crypto_test)
647
648add_executable(ssl_test ${SSL_TEST_SOURCES})
649target_link_libraries(ssl_test test_support_lib boringssl_gtest ssl crypto)
650add_dependencies(all_tests ssl_test)
651add_executable(decrepit_test ${DECREPIT_TEST_SOURCES})
652target_link_libraries(decrepit_test test_support_lib boringssl_gtest
653                      decrepit crypto)
654add_dependencies(all_tests decrepit_test)
655
656if(APPLE)
657  set(PKI_CXX_FLAGS "-fno-aligned-new")
658endif()
659
660add_library(pki ${PKI_SOURCES})
661target_link_libraries(pki crypto)
662
663add_executable(pki_test ${PKI_TEST_SOURCES})
664target_link_libraries(pki_test test_support_lib boringssl_gtest pki crypto)
665add_dependencies(all_tests pki_test)
666
667# The PKI library requires C++17.
668set_target_properties(
669  pki pki_test
670  PROPERTIES
671  CXX_STANDARD 17
672  CXX_STANDARD_REQUIRED YES
673  COMPILE_FLAGS "${PKI_CXX_FLAGS}")
674
675add_executable(bssl ${BSSL_SOURCES})
676target_link_libraries(bssl ssl crypto)
677
678# Historically, targets were built in subdirectories. For compatibility with
679# existing tools, we, for now, copy the targets into the subdirectories. This
680# will be removed sometime in 2024.
681copy_post_build(crypto crypto crypto_test urandom_test)
682copy_post_build(ssl ssl ssl_test)
683copy_post_build(decrepit decrepit decrepit_test)
684copy_post_build(tool bssl)
685
686if(FUZZ)
687  if(LIBFUZZER_FROM_DEPS)
688    file(GLOB LIBFUZZER_SOURCES "util/bot/libFuzzer/*.cpp")
689    add_library(Fuzzer STATIC ${LIBFUZZER_SOURCES})
690    # libFuzzer does not pass our aggressive warnings. It also must be built
691    # without -fsanitize-coverage options or clang crashes.
692    set_target_properties(Fuzzer PROPERTIES COMPILE_FLAGS "-Wno-shadow -Wno-format-nonliteral -Wno-missing-prototypes -fsanitize-coverage=0")
693  endif()
694
695  add_subdirectory(fuzz)
696endif()
697
698if(RUST_BINDINGS)
699  find_program(BINDGEN_EXECUTABLE bindgen)
700  if(NOT BINDGEN_EXECUTABLE)
701    message(FATAL_ERROR "Could not find bindgen but was asked to generate Rust bindings.")
702  else()
703    add_subdirectory(rust)
704  endif()
705endif()
706
707if(CMAKE_SYSTEM_NAME STREQUAL "Linux")
708  set(HANDSHAKER_ARGS "-handshaker-path" $<TARGET_FILE:handshaker>)
709endif()
710
711if(FIPS)
712  add_custom_target(
713    acvp_tests
714    COMMAND ${GO_EXECUTABLE} build -o ${CMAKE_CURRENT_BINARY_DIR}/acvptool
715            boringssl.googlesource.com/boringssl/util/fipstools/acvp/acvptool
716    COMMAND ${GO_EXECUTABLE} build -o ${CMAKE_CURRENT_BINARY_DIR}/testmodulewrapper
717            boringssl.googlesource.com/boringssl/util/fipstools/acvp/acvptool/testmodulewrapper
718    COMMAND cd util/fipstools/acvp/acvptool/test &&
719            ${GO_EXECUTABLE} run check_expected.go
720            -tool ${CMAKE_CURRENT_BINARY_DIR}/acvptool
721            -module-wrappers modulewrapper:$<TARGET_FILE:modulewrapper>,testmodulewrapper:${CMAKE_CURRENT_BINARY_DIR}/testmodulewrapper
722            -tests tests.json
723    WORKING_DIRECTORY ${CMAKE_CURRENT_SOURCE_DIR}
724    DEPENDS modulewrapper
725    USES_TERMINAL)
726
727  add_custom_target(
728    fips_specific_tests_if_any
729    DEPENDS acvp_tests
730  )
731else()
732  add_custom_target(fips_specific_tests_if_any)
733endif()
734
735file(STRINGS util/go_tests.txt GO_TESTS)
736set_property(DIRECTORY APPEND PROPERTY CMAKE_CONFIGURE_DEPENDS
737             util/go_tests.txt)
738
739if(GO_EXECUTABLE)
740  add_custom_target(
741      run_tests
742      COMMAND ${CMAKE_COMMAND} -E echo "Running Go tests"
743      COMMAND ${GO_EXECUTABLE} test ${GO_TESTS}
744      COMMAND ${CMAKE_COMMAND} -E echo
745      COMMAND ${CMAKE_COMMAND} -E echo "Running unit tests"
746      COMMAND ${GO_EXECUTABLE} run util/all_tests.go -build-dir
747              ${CMAKE_CURRENT_BINARY_DIR}
748      COMMAND ${CMAKE_COMMAND} -E echo
749      COMMAND ${CMAKE_COMMAND} -E echo "Running SSL tests"
750      COMMAND cd ssl/test/runner &&
751              ${GO_EXECUTABLE} test -shim-path $<TARGET_FILE:bssl_shim>
752                ${HANDSHAKER_ARGS} ${RUNNER_ARGS}
753      WORKING_DIRECTORY ${CMAKE_CURRENT_SOURCE_DIR}
754      DEPENDS all_tests bssl_shim handshaker fips_specific_tests_if_any
755      USES_TERMINAL)
756else()
757  add_custom_target(
758      run_tests
759      COMMAND ${CMAKE_COMMAND} -E echo "Running tests requires Go"
760      COMMAND ${CMAKE_COMMAND} -E false)
761endif()
762
763if(INSTALL_ENABLED)
764  # CMake versions before 3.14 do not have default destination values. Executable
765  # and library targets that use a default destination should include this
766  # variable.
767  if(CMAKE_VERSION VERSION_LESS "3.14")
768    set(INSTALL_DESTINATION_DEFAULT
769        ARCHIVE DESTINATION ${CMAKE_INSTALL_LIBDIR}
770        LIBRARY DESTINATION ${CMAKE_INSTALL_LIBDIR}
771        RUNTIME DESTINATION ${CMAKE_INSTALL_BINDIR})
772  endif()
773
774  install(TARGETS crypto ssl
775          EXPORT OpenSSLTargets ${INSTALL_DESTINATION_DEFAULT})
776  install(TARGETS bssl DESTINATION ${INSTALL_DESTINATION_DEFAULT})
777  install(DIRECTORY include/ DESTINATION ${CMAKE_INSTALL_INCLUDEDIR})
778  install(EXPORT OpenSSLTargets
779          FILE OpenSSLTargets.cmake
780          NAMESPACE OpenSSL::
781          DESTINATION lib/cmake/OpenSSL)
782  install(FILES cmake/OpenSSLConfig.cmake DESTINATION lib/cmake/OpenSSL)
783endif()
784