| # Copyright (c) 2013 The Chromium Authors. All rights reserved. |
| # Use of this source code is governed by a BSD-style license that can be |
| # found in the LICENSE file. |
| |
| import("//build/config/chrome_build.gni") |
| import("//build/config/c++/c++.gni") |
| import("//build/config/clang/clang.gni") |
| import("//build/config/compiler/compiler.gni") |
| import("//build/config/sanitizers/sanitizers.gni") |
| import("//build/config/win/visual_studio_version.gni") |
| import("//build/toolchain/goma.gni") |
| import("//build/toolchain/toolchain.gni") |
| |
| assert(is_win) |
| |
| declare_args() { |
| # Set this to true to enable static analysis through Visual Studio's |
| # /analyze. This dramatically slows compiles and reports thousands of |
| # warnings, so normally this is done on a build machine and only the new |
| # warnings are examined. |
| use_vs_code_analysis = false |
| |
| # Turn this on to have the linker output extra timing information. |
| win_linker_timing = false |
| |
| # possible values for target_winuwp_version: |
| # "10" - Windows UWP 10 |
| # "8.1" - Windows RT 8.1 |
| # "8.0" - Windows RT 8.0 |
| target_winuwp_version = "10" |
| |
| # possible values: |
| # "app" - Windows Store Applications |
| # "phone" - Windows Phone Applications |
| # "system" - Windows Drivers and Tools |
| # "server" - Windows Server Applications |
| # "desktop" - Windows Desktop Applications |
| target_winuwp_family = "app" |
| } |
| |
| # This is included by reference in the //build/config/compiler config that |
| # is applied to all targets. It is here to separate out the logic that is |
| # Windows-only. |
| config("compiler") { |
| if (current_cpu == "x86") { |
| asmflags = [ |
| # When /safeseh is specified, the linker will only produce an image if it |
| # can also produce a table of the image's safe exception handlers. This |
| # table specifies for the operating system which exception handlers are |
| # valid for the image. Note that /SAFESEH isn't accepted on the command |
| # line, only /safeseh. This is only accepted by ml.exe, not ml64.exe. |
| "/safeseh", |
| ] |
| } |
| |
| cflags = [ |
| "/Gy", # Enable function-level linking. |
| "/FS", # Preserve previous PDB behavior. |
| "/bigobj", # Some of our files are bigger than the regular limits. |
| ] |
| |
| # Force C/C++ mode for the given GN detected file type. This is necessary |
| # for precompiled headers where the same source file is compiled in both |
| # modes. |
| cflags_c = [ "/TC" ] |
| cflags_cc = [ "/TP" ] |
| |
| cflags += [ |
| # Tell the compiler to crash on failures. This is undocumented |
| # and unsupported but very handy. |
| "/d2FastFail", |
| |
| # Work around crbug.com/526851, bug in VS 2015 RTM compiler. |
| "/Zc:sizedDealloc-", |
| ] |
| |
| if (is_clang) { |
| cflags += [ "-fmsc-version=1911" ] |
| |
| if (current_cpu == "x86") { |
| cflags += [ "-m32" ] |
| } else { |
| cflags += [ "-m64" ] |
| } |
| |
| if (exec_script("//build/win/use_ansi_codes.py", [], "trim string") == |
| "True") { |
| cflags += [ |
| # cmd.exe doesn't understand ANSI escape codes by default, |
| # so only enable them if something emulating them is around. |
| "-fansi-escape-codes", |
| ] |
| } |
| |
| # Clang runtime libraries, such as the sanitizer runtimes, live here. |
| lib_dirs = [ "$clang_base_path/lib/clang/$clang_version/lib/windows" ] |
| } |
| |
| if (use_lld && !use_thin_lto && (is_clang || !use_goma)) { |
| # /Brepro lets the compiler not write the mtime field in the .obj output. |
| # link.exe /incremental relies on this field to work correctly, but lld |
| # never looks at this timestamp, so it's safe to pass this flag with |
| # lld and get more deterministic compiler output in return. |
| # In LTO builds, the compiler doesn't write .obj files containing mtimes, |
| # so /Brepro is ignored there. |
| cflags += [ "/Brepro" ] |
| } |
| |
| if (!is_debug && !is_component_build) { |
| # Enable standard linker optimizations like GC (/OPT:REF) and ICF in static |
| # release builds. These are implied by /PROFILE below, but /PROFILE is |
| # incompatible with /debug:fastlink and LLD ignores it as of this writing. |
| # Release builds always want these optimizations, so enable them explicitly. |
| ldflags = [ |
| "/OPT:REF", |
| "/OPT:ICF", |
| "/INCREMENTAL:NO", |
| "/FIXED:NO", |
| ] |
| |
| # TODO(siggi): Is this of any use anymore? |
| # /PROFILE ensures that the PDB file contains FIXUP information (growing the |
| # PDB file by about 5%) but does not otherwise alter the output binary. It |
| # is enabled opportunistically for builds where it is not prohibited (not |
| # supported when incrementally linking, or using /debug:fastlink). |
| if (!is_win_fastlink) { |
| ldflags += [ "/PROFILE" ] |
| } |
| } |
| |
| # arflags apply only to static_libraries. The normal linker configs are only |
| # set for executable and shared library targets so arflags must be set |
| # elsewhere. Since this is relatively contained, we just apply them in this |
| # more general config and they will only have an effect on static libraries. |
| arflags = [ |
| # "No public symbols found; archive member will be inaccessible." This |
| # means that one or more object files in the library can never be |
| # pulled in to targets that link to this library. It's just a warning that |
| # the source file is a no-op. |
| "/ignore:4221", |
| ] |
| } |
| |
| config("vs_code_analysis") { |
| if (use_vs_code_analysis && !is_clang) { |
| # When use_vs_code_analysis is specified add the /analyze switch to enable |
| # static analysis. Specifying /analyze:WX- says that /analyze warnings |
| # should not be treated as errors. |
| cflags = [ "/analyze:WX-" ] |
| |
| # Also, disable various noisy warnings that have low value. |
| cflags += [ |
| "/wd6011", # Dereferencing NULL pointer |
| |
| # C6285 is ~16% of raw warnings and has low value |
| "/wd6285", # non-zero constant || non-zero constant |
| "/wd6308", # realloc might return null pointer |
| |
| # Possible infinite loop: use of the constant |
| # EXCEPTION_CONTINUE_EXECUTION in the exception-filter |
| "/wd6312", |
| |
| "/wd6322", # Empty _except block |
| "/wd6330", # 'char' used instead of 'unsigned char' for istype() call |
| |
| # C6334 is ~80% of raw warnings and has low value |
| "/wd6334", # sizeof applied to an expression with an operator |
| "/wd6326", # Potential comparison of constant with constant |
| "/wd6340", # Sign mismatch in function parameter |
| "/wd28159", # Consider using 'GetTickCount64' |
| "/wd28196", # The precondition is not satisfied |
| "/wd28204", # Inconsistent SAL annotations |
| "/wd28251", # Inconsistent SAL annotations |
| "/wd28252", # Inconsistent SAL annotations |
| "/wd28253", # Inconsistent SAL annotations |
| "/wd28278", # Function appears with no prototype in scope |
| "/wd28285", # syntax error in SAL annotation (in algorithm) |
| "/wd28301", # Inconsistent SAL annotations |
| "/wd28182", # Dereferencing NULL pointer |
| ] |
| } |
| } |
| |
| # This is included by reference in the //build/config/compiler:runtime_library |
| # config that is applied to all targets. It is here to separate out the logic |
| # that is Windows-only. Please see that target for advice on what should go in |
| # :runtime_library vs. :compiler. |
| config("runtime_library") { |
| cflags = [] |
| |
| # Defines that set up the CRT. |
| defines = [ |
| "__STD_C", |
| "_CRT_RAND_S", |
| "_CRT_SECURE_NO_DEPRECATE", |
| "_SCL_SECURE_NO_DEPRECATE", |
| ] |
| |
| if (is_clang) { |
| # Work around Fall Creators Update SDK bug - crbug.com/773476 has details. |
| # https://developercommunity.visualstudio.com/content/problem/131391/154-fails-to-define-deprecatedenumerator-2.html |
| defines += [ "DEPRECATEDENUMERATOR(x)=[[deprecated(x)]]" ] |
| } |
| |
| # Defines that set up the Windows SDK. |
| defines += [ |
| "_ATL_NO_OPENGL", |
| "_WINDOWS", |
| "CERT_CHAIN_PARA_HAS_EXTRA_FIELDS", |
| "PSAPI_VERSION=1", |
| "WIN32", |
| "_SECURE_ATL", |
| ] |
| |
| if (!use_vs_code_analysis) { |
| # This is required for ATL to use XP-safe versions of its functions. |
| # However it is prohibited when using /analyze |
| defines += [ "_USING_V110_SDK71_" ] |
| } |
| |
| if (use_custom_libcxx) { |
| cflags_cc = [ "-I" + rebase_path("$libcxx_prefix/include", root_build_dir) ] |
| if (libcpp_is_static) { |
| defines += [ "_LIBCPP_DISABLE_VISIBILITY_ANNOTATIONS" ] |
| } |
| |
| # Prevent libc++ from embedding linker flags to try to automatically link |
| # against its runtime library. This is unnecessary with our build system, |
| # and can also result in build failures if libc++'s name for a library does |
| # not match ours. |
| defines += [ "_LIBCPP_NO_AUTO_LINK" ] |
| } |
| |
| if (current_os == "winuwp") { |
| # When targeting Windows Runtime, certain compiler/linker flags are |
| # necessary. |
| defines += [ |
| "WINUWP", |
| "__WRL_NO_DEFAULT_LIB__", |
| ] |
| if (target_winuwp_family == "app") { |
| defines += [ "WINAPI_FAMILY=WINAPI_FAMILY_PC_APP" ] |
| } else if (target_winuwp_family == "phone") { |
| defines += [ "WINAPI_FAMILY=WINAPI_FAMILY_PHONE_APP" ] |
| } else if (target_winuwp_family == "system") { |
| defines += [ "WINAPI_FAMILY=WINAPI_FAMILY_SYSTEM" ] |
| } else if (target_winuwp_family == "server") { |
| defines += [ "WINAPI_FAMILY=WINAPI_FAMILY_SERVER" ] |
| } else { |
| defines += [ "WINAPI_FAMILY=WINAPI_FAMILY_DESKTOP_APP" ] |
| } |
| cflags_cc += [ |
| "/ZW", |
| "/EHsc", |
| ] |
| |
| # This warning is given because the linker cannot tell the difference |
| # between consuming WinRT APIs versus authoring WinRT within static |
| # libraries as such this warning is always given by the linker. Since |
| # consuming WinRT APIs within a library is legitimate but authoring |
| # WinRT APis is not allowed, this warning is disabled to ignore the |
| # legitimate consumption of WinRT APIs within static library builds. |
| arflags = [ "/IGNORE:4264" ] |
| |
| if (target_winuwp_version == "10") { |
| defines += [ "WIN10=_WIN32_WINNT_WIN10" ] |
| } else if (target_winuwp_version == "8.1") { |
| defines += [ "WIN8_1=_WIN32_WINNT_WINBLUE" ] |
| } else if (target_winuwp_version == "8.0") { |
| defines += [ "WIN8=_WIN32_WINNT_WIN8" ] |
| } |
| } else { |
| # When not targeting Windows Runtime, make sure the WINAPI family is set |
| # to desktop. |
| defines += [ "WINAPI_FAMILY=WINAPI_FAMILY_DESKTOP_APP" ] |
| } |
| } |
| |
| # Sets the default Windows build version. This is separated because some |
| # targets need to manually override it for their compiles. |
| config("winver") { |
| defines = [ |
| "NTDDI_VERSION=0x0A000002", |
| "_WIN32_WINNT=0x0A00", |
| "WINVER=0x0A00", |
| ] |
| } |
| |
| # Linker flags for Windows SDK setup, this is applied only to EXEs and DLLs. |
| config("sdk_link") { |
| assert(current_cpu == "x64" || current_cpu == "x86" || current_cpu == "arm" || |
| current_cpu == "arm64", |
| "Only supports x64, x86, arm and arm64 CPUs") |
| if (current_cpu == "x64") { |
| ldflags = [ "/MACHINE:X64" ] |
| } else if (current_cpu == "x86") { |
| ldflags = [ |
| "/MACHINE:X86", |
| "/SAFESEH", # Not compatible with x64 so use only for x86. |
| "/largeaddressaware", |
| ] |
| } else if (current_cpu == "arm") { |
| ldflags = [ "/MACHINE:ARM" ] |
| } else if (current_cpu == "arm64") { |
| ldflags = [ "/MACHINE:ARM64" ] |
| } |
| |
| vcvars_toolchain_data = exec_script("../../toolchain/win/setup_toolchain.py", |
| [ |
| visual_studio_path, |
| windows_sdk_path, |
| visual_studio_runtime_dirs, |
| current_os, |
| current_cpu, |
| "none", |
| ], |
| "scope") |
| |
| vc_lib_path = vcvars_toolchain_data.vc_lib_path |
| if (defined(vcvars_toolchain_data.vc_lib_atlmfc_path)) { |
| vc_lib_atlmfc_path = vcvars_toolchain_data.vc_lib_atlmfc_path |
| } |
| vc_lib_um_path = vcvars_toolchain_data.vc_lib_um_path |
| |
| lib_dirs = [ |
| "$vc_lib_um_path", |
| "$vc_lib_path", |
| ] |
| if (defined(vc_lib_atlmfc_path)) { |
| lib_dirs += [ "$vc_lib_atlmfc_path" ] |
| } |
| } |
| |
| # This default linker setup is provided separately from the SDK setup so |
| # targets who want different library configurations can remove this and specify |
| # their own. |
| config("common_linker_setup") { |
| ldflags = [ |
| "/fastfail", |
| "/FIXED:NO", |
| "/ignore:4199", |
| "/ignore:4221", |
| "/NXCOMPAT", |
| "/DYNAMICBASE", |
| ] |
| |
| if (win_linker_timing) { |
| ldflags += [ |
| "/time", |
| "/verbose:incr", |
| ] |
| } |
| } |
| |
| config("cfi_linker") { |
| # Control Flow Guard (CFG) |
| # https://msdn.microsoft.com/en-us/library/windows/desktop/mt637065.aspx |
| # /DYNAMICBASE (ASLR) is turned off in debug builds, therefore CFG can’t be |
| # turned on either. |
| # CFG seems to lead to random corruption with incremental linking so turn off |
| # CFG in component builds. https://crbug.com/812421 |
| if (!is_debug && !is_component_build) { |
| # Turn on CFG in msvc linker, regardless of compiler used. Turn off CFG for |
| # longjmp (new in VS 2017) because it relies on compiler support which we do |
| # not have enabled. |
| ldflags = [ "/guard:cf,nolongjmp" ] |
| } |
| } |
| |
| # CRT -------------------------------------------------------------------------- |
| |
| # Configures how the runtime library (CRT) is going to be used. |
| # See https://msdn.microsoft.com/en-us/library/2kzt1wy3.aspx for a reference of |
| # what each value does. |
| config("default_crt") { |
| if (is_component_build) { |
| # Component mode: dynamic CRT. Since the library is shared, it requires |
| # exceptions or will give errors about things not matching, so keep |
| # exceptions on. |
| configs = [ ":dynamic_crt" ] |
| } else { |
| if (current_os == "winuwp") { |
| # https://blogs.msdn.microsoft.com/vcblog/2014/06/10/the-great-c-runtime-crt-refactoring/ |
| # contains a details explanation of what is happening with the Windows |
| # CRT in Visual Studio releases related to Windows store applications. |
| configs = [ ":dynamic_crt" ] |
| } else { |
| # Desktop Windows: static CRT. |
| configs = [ ":static_crt" ] |
| } |
| } |
| } |
| |
| # Use this to force the debug CRT for when building perf-critical build tools |
| # that need to be fully optimized even in debug builds, for those times when the |
| # debug CRT is part of the bottleneck. This also avoids *implicitly* defining |
| # _DEBUG. |
| config("release_crt") { |
| if (is_component_build) { |
| cflags = [ "/MD" ] |
| } else { |
| cflags = [ "/MT" ] |
| } |
| } |
| |
| config("dynamic_crt") { |
| if (is_debug) { |
| # This pulls in the DLL debug CRT and defines _DEBUG |
| cflags = [ "/MDd" ] |
| } else { |
| cflags = [ "/MD" ] |
| } |
| } |
| |
| config("static_crt") { |
| if (is_debug) { |
| # This pulls in the static debug CRT and defines _DEBUG |
| cflags = [ "/MTd" ] |
| } else { |
| cflags = [ "/MT" ] |
| } |
| } |
| |
| # Subsystem -------------------------------------------------------------------- |
| |
| # This is appended to the subsystem to specify a minimum version. |
| if (current_cpu == "x64") { |
| # The number after the comma is the minimum required OS version. |
| # 5.02 = Windows Server 2003. |
| subsystem_version_suffix = ",5.02" |
| } else { |
| # 5.01 = Windows XP. |
| subsystem_version_suffix = ",5.01" |
| } |
| |
| config("console") { |
| ldflags = [ "/SUBSYSTEM:CONSOLE$subsystem_version_suffix" ] |
| } |
| config("windowed") { |
| ldflags = [ "/SUBSYSTEM:WINDOWS$subsystem_version_suffix" ] |
| } |
| |
| # Incremental linking ---------------------------------------------------------- |
| |
| incremental_linking_on_switch = [ "/INCREMENTAL" ] |
| incremental_linking_off_switch = [ "/INCREMENTAL:NO" ] |
| if (use_lld) { |
| incremental_linking_on_switch += [ "/OPT:NOREF" ] |
| } |
| |
| # Enable incremental linking for debug builds and all component builds - any |
| # builds where performance is not job one. |
| if (is_debug || is_component_build) { |
| default_incremental_linking_switch = incremental_linking_on_switch |
| } else { |
| default_incremental_linking_switch = incremental_linking_off_switch |
| } |
| |
| # Applies incremental linking or not depending on the current configuration. |
| config("default_incremental_linking") { |
| ldflags = default_incremental_linking_switch |
| } |
| |
| # Explicitly on or off incremental linking |
| config("incremental_linking") { |
| ldflags = incremental_linking_on_switch |
| } |
| config("no_incremental_linking") { |
| ldflags = incremental_linking_off_switch |
| } |
| |
| # Some large modules can't handle incremental linking in some situations. This |
| # config should be applied to large modules to turn off incremental linking |
| # when it won't work. |
| config("default_large_module_incremental_linking") { |
| if (symbol_level == 0 || (current_cpu == "x86" && is_component_build)) { |
| # In these configurations, ilk file sizes stay low enough that we can |
| # link incrementally. |
| ldflags = default_incremental_linking_switch |
| } else { |
| ldflags = incremental_linking_off_switch |
| } |
| } |
| |
| # Character set ---------------------------------------------------------------- |
| |
| # Not including this config means "ansi" (8-bit system codepage). |
| config("unicode") { |
| defines = [ |
| "_UNICODE", |
| "UNICODE", |
| ] |
| } |
| |
| # Lean and mean ---------------------------------------------------------------- |
| |
| # Some third party code might not compile with WIN32_LEAN_AND_MEAN so we have |
| # to have a separate config for it. Remove this config from your target to |
| # get the "bloaty and accommodating" version of windows.h. |
| config("lean_and_mean") { |
| defines = [ "WIN32_LEAN_AND_MEAN" ] |
| } |
| |
| # Nominmax -------------------------------------------------------------------- |
| |
| # Some third party code defines NOMINMAX before including windows.h, which |
| # then causes warnings when it's been previously defined on the command line. |
| # For such targets, this config can be removed. |
| |
| config("nominmax") { |
| defines = [ "NOMINMAX" ] |
| } |
| |
| # Generating order files ------------------------------------------------------- |
| |
| config("default_cygprofile_instrumentation") { |
| if (generate_order_files) { |
| assert(is_clang, "cygprofile instrumentation only works with clang") |
| assert(is_official_build, "order files should be made w/ official builds") |
| assert(!is_chrome_branded, "order files could leak internal symbol names") |
| configs = [ ":cygprofile_instrumentation" ] |
| } |
| } |
| |
| config("cygprofile_instrumentation") { |
| cflags = [ |
| "-Xclang", |
| "-finstrument-functions-after-inlining", |
| ] |
| } |