1# Copyright 2015 The Chromium Authors 2# Use of this source code is governed by a BSD-style license that can be 3# found in the LICENSE file. 4 5import("//build/config/cast.gni") 6import("//build/config/chrome_build.gni") 7import("//build/config/chromeos/args.gni") 8import("//build/config/chromeos/ui_mode.gni") 9import("//build/config/profiling/profiling.gni") 10import("//build/toolchain/toolchain.gni") 11 12declare_args() { 13 # Compile for Address Sanitizer to find memory bugs. 14 is_asan = false 15 16 # Compile for Hardware-Assisted Address Sanitizer to find memory bugs 17 # (android/arm64 only). 18 # See http://clang.llvm.org/docs/HardwareAssistedAddressSanitizerDesign.html 19 is_hwasan = false 20 21 # Compile for Leak Sanitizer to find leaks. 22 is_lsan = false 23 24 # Compile for Memory Sanitizer to find uninitialized reads. 25 is_msan = false 26 27 # Compile for Thread Sanitizer to find threading bugs. 28 is_tsan = false 29 30 # Compile for Undefined Behaviour Sanitizer to find various types of 31 # undefined behaviour (excludes vptr checks). 32 is_ubsan = false 33 34 # Halt the program if a problem is detected. 35 is_ubsan_no_recover = false 36 37 # Track where uninitialized memory originates from. From fastest to slowest: 38 # 0 - no tracking, 1 - track only the initial allocation site, 2 - track the 39 # chain of stores leading from allocation site to use site. 40 msan_track_origins = 2 41 42 # Enables "param-retval" mode, which finds more uses of uninitialized data and 43 # reduces code size. Behind a flag as there are a number of previously 44 # undetected violations that still need to be fixed. 45 # TODO(crbug.com/40240570): Default this to true and remove. 46 msan_eager_checks = true 47 48 # TODO(crbug.com/40222690): Enable everywhere. 49 msan_check_use_after_dtor = is_linux 50 51 # Use dynamic libraries instrumented by one of the sanitizers instead of the 52 # standard system libraries. Set this flag to build the libraries from source. 53 use_locally_built_instrumented_libraries = false 54 55 # Compile with Control Flow Integrity to protect virtual calls and casts. 56 # See http://clang.llvm.org/docs/ControlFlowIntegrity.html 57 # 58 # TODO(pcc): Remove this flag if/when CFI is enabled in all official builds. 59 is_cfi = is_official_build && is_clang && 60 ((target_os == "linux" && target_cpu == "x64") || 61 (is_chromeos && is_chromeos_device)) 62 63 # Enable checks for indirect function calls via a function pointer. 64 # TODO(pcc): remove this when we're ready to add these checks by default. 65 # https://crbug.com/701919 66 use_cfi_icall = 67 target_os == "linux" && target_cpu == "x64" && is_official_build 68 69 # Print detailed diagnostics when Control Flow Integrity detects a violation. 70 use_cfi_diag = false 71 72 # Let Control Flow Integrity continue execution instead of crashing when 73 # printing diagnostics (use_cfi_diag = true). 74 use_cfi_recover = false 75 76 # Compile for fuzzing with LLVM LibFuzzer. 77 # See http://www.chromium.org/developers/testing/libfuzzer 78 use_libfuzzer = false 79 80 # Compile for fuzzing with centipede. 81 # See https://github.com/google/centipede 82 use_centipede = false 83 84 # Compile for fuzzing with AFL. 85 use_afl = false 86 87 # Compile for fuzzing with Fuzzilli. 88 use_fuzzilli = false 89 90 # Compile for fuzzing with an external engine (e.g., Grammarinator). 91 use_external_fuzzing_engine = false 92 93 # Enables core ubsan security features. Will later be removed once it matches 94 # is_ubsan. 95 is_ubsan_security = false 96 97 # Helper variable for testing builds with disabled libfuzzer. 98 # Not for client use. 99 disable_libfuzzer = false 100 101 # Value for -fsanitize-coverage flag. Setting this causes 102 # use_sanitizer_coverage to be enabled. 103 # This flag is not used for libFuzzer (use_libfuzzer=true). Instead, we use: 104 # -fsanitize=fuzzer-no-link 105 # Default value when unset and use_fuzzing_engine=true: 106 # trace-pc-guard 107 # Default value when unset and use_sanitizer_coverage=true: 108 # trace-pc-guard,indirect-calls 109 sanitizer_coverage_flags = "" 110 111 # A sanitizer coverage allowlist, specifying exactly which 112 # files or symbol names should be instrumented, rather than all of them. 113 sanitizer_coverage_allowlist = "" 114 115 # When enabled, only relevant sanitizer defines are set, but compilation 116 # happens with no extra flags. This is useful when in component build 117 # enabling sanitizers only in some of the components. 118 use_sanitizer_configs_without_instrumentation = false 119 120 # When true, seed corpora archives are built. 121 archive_seed_corpus = true 122 123 # When true, only builds fuzzer targets that require high end machines to run. 124 # Otherwise, builds all the targets. 125 # TODO(paulsemel): once we have everything implemented on the recipe side, we 126 # can change the behaviour for the false case, and only build the non high-end 127 # jobs, so that they do not appear in the zip. As for now, this behaviour 128 # ensures nothing breaks. 129 high_end_fuzzer_targets = false 130} 131 132declare_args() { 133 # Enable checks for bad casts: derived cast and unrelated cast. 134 # TODO(krasin): remove this, when we're ready to add these checks by default. 135 # https://crbug.com/626794 136 use_cfi_cast = is_cfi && is_chromeos 137 138 # Compile for Undefined Behaviour Sanitizer's vptr checks. 139 is_ubsan_vptr = is_ubsan_security || is_ubsan 140 141 # These are set for some non-host toolchain which requires to run built binary 142 # from host toolchain with msan config but in non-host toolchain context. 143 host_toolchain_is_msan = is_msan 144 host_toolchain_msan_track_origins = msan_track_origins 145} 146 147# Whether we are doing a fuzzer build. Normally this should be checked instead 148# of checking "use_libfuzzer || use_afl" because often developers forget to 149# check for "use_afl", and "use_centipede" is new. 150use_fuzzing_engine = use_libfuzzer || use_afl || use_centipede || 151 use_external_fuzzing_engine || use_fuzzilli 152 153# Whether any UBSan subset is enabled. 154# TODO(crbug.com/40248746): Unify these under a single UBSan option. 155is_ubsan_any = is_ubsan || is_ubsan_vptr || is_ubsan_security 156 157declare_args() { 158 # Builds fuzztest test executables such that they support the 159 # --fuzz= argument, which requires some sanitizer coverage. 160 # We want to enable this only when we're *NOT* using a fuzzing 161 # engine such as libfuzzer or centipede. It's generally a 162 # useful option, but it requires sanitizer coverage, and that 163 # could conceivably disrupt normal unit testing workflows, so we'll 164 # enable it by default only in sanitizer builds. 165 # Also be sure not to enable this on non-Chromium builds where 166 # the required //third_party/fuzztest dependency may be absent. 167 # TODO(crbug.com/40286621): enable on component builds 168 enable_fuzztest_fuzz = 169 !(use_libfuzzer || use_afl || use_centipede || 170 use_external_fuzzing_engine || use_fuzzilli) && 171 (is_asan || is_hwasan || is_lsan || is_tsan || is_msan || is_ubsan_any) && 172 !is_component_build && is_linux && build_with_chromium 173 174 # Whether FUZZ_TESTs should be registered as gtests in Chromium's mainstream 175 # test suites. This will cause the to run for 1 second in normal testing 176 # environments such as CI. Within Chromium, we want to do this only on 177 # platforms where we have ClusterFuzz support (Windows, Mac, Linux) because 178 # we want ClusterFuzz to find the majority of problems rather than CI. 179 # This gn arg is typically respected by general-purpose test suite 180 # initialization code; targets created specifically for fuzztests alone 181 # would not normally be disabled by a setting of 'false' here. 182 register_fuzztests_in_test_suites = 183 is_linux || is_mac || is_win || use_libfuzzer || use_afl || 184 use_centipede || use_external_fuzzing_engine 185} 186 187declare_args() { 188 use_sanitizer_coverage = 189 !use_clang_coverage && (use_fuzzing_engine || enable_fuzztest_fuzz || 190 sanitizer_coverage_flags != "") 191} 192 193assert(!is_hwasan || (target_os == "android" && target_cpu == "arm64"), 194 "HWASan only supported on Android ARM64 builds.") 195 196assert( 197 !(enable_fuzztest_fuzz && use_libfuzzer), 198 "Can't specify enable_fuzztest_fuzz and use_libfuzzer. When libfuzzer is enabled, fuzztest executables automatically support --fuzz but provide output that's libfuzzer compatible.") 199 200assert( 201 !(enable_fuzztest_fuzz && use_centipede), 202 "Can't specify enable_fuzztest_fuzz and use_centipede. The same binaries are built in a different mode to add centipede support.") 203 204assert( 205 !(enable_fuzztest_fuzz && is_component_build), 206 "Can't specify enable_fuzztest_fuzz in component builds; fuzztest doesn't yet support it. Consider using use_libfuzzer=true instead which provides superficially similar functionality.") 207 208# Disable sanitizers for non-target toolchains, and for the toolchain using 209# the prebuilt Rust stdlib which has no sanitizer support with it. 210if (!is_a_target_toolchain || toolchain_for_rust_host_build_tools) { 211 is_asan = false 212 is_cfi = false 213 is_hwasan = false 214 is_lsan = false 215 is_msan = false 216 is_tsan = false 217 is_ubsan = false 218 is_ubsan_no_recover = false 219 is_ubsan_security = false 220 is_ubsan_vptr = false 221 is_ubsan_any = false 222 msan_track_origins = 0 223 sanitizer_coverage_flags = "" 224 use_afl = false 225 use_centipede = false 226 use_cfi_diag = false 227 use_cfi_recover = false 228 use_libfuzzer = false 229 use_locally_built_instrumented_libraries = false 230 use_sanitizer_coverage = false 231 enable_fuzztest_fuzz = false 232} else if (current_cpu != "arm64") { 233 is_hwasan = false 234} 235 236# Whether we are linking against a sanitizer runtime library. Among other 237# things, this changes the default symbol level and other settings in order to 238# prepare to create stack traces "live" using the sanitizer runtime. 239using_sanitizer = is_asan || is_hwasan || is_lsan || is_tsan || is_msan || 240 is_ubsan_any || use_sanitizer_coverage || use_cfi_diag 241 242# Use dynamic libraries instrumented by one of the sanitizers instead of the 243# standard system libraries. We have instrumented system libraries for msan, 244# which requires them to prevent false positives. 245# TODO(thakis): Maybe remove this variable. 246use_prebuilt_instrumented_libraries = is_msan 247 248# Whether the current fuzzing engine supports libprotobuf_mutator. 249use_fuzzing_engine_with_lpm = use_libfuzzer || use_centipede 250 251# Whether the fuzzing engine supports fuzzers which supply their own 252# "main" function. 253fuzzing_engine_supports_custom_main = 254 use_libfuzzer || use_centipede || use_fuzzilli 255 256# Args that are in turn dependent on other args must be in a separate 257# declare_args block. User overrides are only applied at the end of a 258# declare_args block. 259declare_args() { 260 # When true, sanitizer warnings will cause test case failures. 261 fail_on_san_warnings = using_sanitizer 262 263 # Generates an owners file for each fuzzer test. 264 # TODO(crbug.com/40175535): Remove this arg when finding OWNERS is faster. 265 generate_fuzzer_owners = use_fuzzing_engine 266 267 # https://crbug.com/1002058: Code coverage works inside the sandbox via the 268 # help of several helper IPCs. Unfortunately, the sandbox-only path does not 269 # work well for fuzzing builds. Since fuzzing builds already disable the 270 # sandbox when dumping coverage, limit the sandbox-only path to non-fuzzing 271 # builds. 272 # Everything is IPC on Fuchsia, so this workaround for code coverage inside 273 # the sandbox does not apply. 274 use_clang_profiling_inside_sandbox = 275 use_clang_profiling && !use_fuzzing_engine && !is_fuchsia 276} 277 278if (sanitizer_coverage_flags == "") { 279 if (enable_fuzztest_fuzz) { 280 # ./fuzztest_executable --fuzz= 281 # requires only this single type of coverage 282 sanitizer_coverage_flags = "inline-8bit-counters" 283 } else if (use_fuzzing_engine) { 284 sanitizer_coverage_flags = "trace-pc-guard" 285 if (use_centipede) { 286 # Centipede's minimal flags are listed in //third_party/fuzztest/src/centipede/clang-flags.txt. 287 # But, for users like Chromium using an up-to-date clang, we can also 288 # enable extra optional types of coverage which may make Centipede more 289 # effective. This list is not currently documented and has been derived 290 # from discussion with centipede creators (though one is warned about at 291 # https://github.com/google/centipede/blob/main/centipede_callbacks.cc#L68) 292 sanitizer_coverage_flags = sanitizer_coverage_flags + 293 ",pc-table,trace-cmp,control-flow,trace-loads" 294 } 295 } else if (use_sanitizer_coverage) { 296 sanitizer_coverage_flags = "trace-pc-guard,indirect-calls" 297 } 298} 299 300assert(!using_sanitizer || is_clang, 301 "Sanitizers (is_*san) require setting is_clang = true in 'gn args'") 302 303assert(!is_cfi || is_clang, 304 "is_cfi requires setting is_clang = true in 'gn args'") 305 306prebuilt_instrumented_libraries_available = 307 is_msan && (msan_track_origins == 0 || msan_track_origins == 2) 308 309if (use_libfuzzer && (is_linux || is_chromeos)) { 310 if (is_asan) { 311 # We do leak checking with libFuzzer on Linux. Set is_lsan for code that 312 # relies on LEAK_SANITIZER define to avoid false positives. 313 is_lsan = true 314 } 315} 316 317# MSan only links Chrome properly in release builds (brettw -- 9/1/2015). The 318# same is possibly true for the other non-ASan sanitizers. But regardless of 319# whether it links, one would normally never run a sanitizer in debug mode. 320# Running in debug mode probably indicates you forgot to set the "is_debug = 321# false" flag in the build args. ASan seems to run fine in debug mode. 322# 323# If you find a use-case where you want to compile a sanitizer in debug mode 324# and have verified it works, ask brettw and we can consider removing it from 325# this condition. We may also be able to find another way to enable your case 326# without having people accidentally get broken builds by compiling an 327# unsupported or unadvisable configurations. 328# 329# For one-off testing, just comment this assertion out. 330assert(!is_debug || !(is_msan || is_ubsan || is_ubsan_vptr), 331 "Sanitizers should generally be used in release (set is_debug=false).") 332 333assert(!is_msan || ((is_linux || is_chromeos) && current_cpu == "x64"), 334 "MSan currently only works on 64-bit Linux and ChromeOS builds.") 335 336assert(!is_lsan || is_asan, "is_lsan = true requires is_asan = true also.") 337 338# ASAN build on Windows is not working in debug mode. Intercepting memory 339# allocation functions is hard on Windows and not yet implemented in LLVM. 340assert(!is_win || !is_debug || !is_asan, 341 "ASan on Windows doesn't work in debug (set is_debug=false).") 342 343# libFuzzer targets can fail to build or behave incorrectly when built without 344# ASAN on Windows. 345assert(!is_win || !use_libfuzzer || is_asan, 346 "use_libfuzzer on Windows requires setting is_asan = true") 347 348# Make sure that if we recover on detection (i.e. not crash), diagnostics are 349# printed. 350assert(!use_cfi_recover || use_cfi_diag, 351 "Only use CFI recovery together with diagnostics.") 352 353# TODO(crbug.com/40534102): the use_sanitizer_coverage arg is currently 354# not supported by the Chromium mac_clang_x64 toolchain on iOS distribution. 355# The coverage works with iOS toolchain but it is broken when the mac 356# toolchain is used as a secondary one on iOS distribution. E.g., it should be 357# possible to build the "net" target for iOS with the sanitizer coverage 358# enabled. 359assert( 360 !(use_sanitizer_coverage && is_mac && target_os == "ios"), 361 "crbug.com/753445: use_sanitizer_coverage=true is not supported by the " + 362 "Chromium mac_clang_x64 toolchain on iOS distribution. Please set " + 363 "the argument value to false.") 364 365assert( 366 sanitizer_coverage_allowlist == "" || use_sanitizer_coverage, 367 "Can't specify a sanitizer coverage allowlist without using sanitizer coverage.") 368 369# Use these lists of configs to disable instrumenting code that is part of a 370# fuzzer, but which isn't being targeted (such as libprotobuf-mutator, *.pb.cc 371# and libprotobuf when they are built as part of a proto fuzzer). Adding or 372# removing these lists does not have any effect if use_libfuzzer or use_afl are 373# not passed as arguments to gn. 374not_fuzzed_remove_configs = [] 375not_fuzzed_remove_nonasan_configs = [] 376 377if (use_fuzzing_engine) { 378 # Removing coverage should always just work. 379 not_fuzzed_remove_configs += [ "//build/config/coverage:default_coverage" ] 380 not_fuzzed_remove_nonasan_configs += 381 [ "//build/config/coverage:default_coverage" ] 382 383 if (!is_msan) { 384 # Allow sanitizer instrumentation to be removed if we are not using MSan 385 # since binaries cannot be partially instrumented with MSan. 386 not_fuzzed_remove_configs += 387 [ "//build/config/sanitizers:default_sanitizer_flags" ] 388 389 # Certain parts of binaries must be instrumented with ASan if the rest of 390 # the binary is. For these, only remove non-ASan sanitizer instrumentation. 391 if (!is_asan) { 392 not_fuzzed_remove_nonasan_configs += 393 [ "//build/config/sanitizers:default_sanitizer_flags" ] 394 395 assert(not_fuzzed_remove_nonasan_configs == not_fuzzed_remove_configs) 396 } 397 } 398} 399 400# Options common to different fuzzer engines. 401# Engine should be compiled without coverage (infinite loop in trace_cmp). 402fuzzing_engine_remove_configs = [ 403 "//build/config/coverage:default_coverage", 404 "//build/config/sanitizers:default_sanitizer_flags", 405] 406 407# Add any sanitizer flags back. In MSAN builds, instrumenting libfuzzer with 408# MSAN is necessary since all parts of the binary need to be instrumented for it 409# to work. ASAN builds are more subtle: libfuzzer depends on features from the 410# C++ STL. If it were not instrumented, templates would be insantiated without 411# ASAN from libfuzzer and with ASAN in other TUs. The linker might merge 412# instrumented template instantiations with non-instrumented ones (which could 413# have a different ABI) in the final binary, which is problematic for TUs 414# expecting one particular ABI (https://crbug.com/915422). The other sanitizers 415# are added back for the same reason. 416fuzzing_engine_add_configs = 417 [ "//build/config/sanitizers:default_sanitizer_flags_but_coverage" ] 418