2021-04-15 10:10:20 +00:00
|
|
|
#!/usr/bin/env python3
|
2016-07-01 18:09:28 +00:00
|
|
|
# Copyright 2014 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.
|
|
|
|
|
2017-09-28 09:25:37 +00:00
|
|
|
# For Dart developers:
|
|
|
|
# This file keeps the MSVC toolchain up-to-date for Google developers.
|
|
|
|
# It is copied from Chromium:
|
|
|
|
# https://cs.chromium.org/chromium/src/build/vs_toolchain.py
|
2019-07-31 20:01:01 +00:00
|
|
|
# with modifications that update paths and remove dependencies on gyp.
|
2017-09-28 09:25:37 +00:00
|
|
|
# To update to a new MSVC toolchain, copy the updated script from the Chromium
|
2019-07-31 20:01:01 +00:00
|
|
|
# tree, edit to make it work in the Dart tree by updating paths in the original script.
|
2017-09-28 09:25:37 +00:00
|
|
|
|
2019-07-31 20:01:01 +00:00
|
|
|
from __future__ import print_function
|
|
|
|
|
|
|
|
import collections
|
2016-10-04 16:19:07 +00:00
|
|
|
import glob
|
2016-07-01 18:09:28 +00:00
|
|
|
import json
|
|
|
|
import os
|
|
|
|
import pipes
|
2016-10-04 16:19:07 +00:00
|
|
|
import platform
|
2017-09-28 09:25:37 +00:00
|
|
|
import re
|
2016-07-01 18:09:28 +00:00
|
|
|
import shutil
|
2016-10-04 16:19:07 +00:00
|
|
|
import stat
|
2016-07-01 18:09:28 +00:00
|
|
|
import subprocess
|
|
|
|
import sys
|
2018-11-18 22:06:49 +00:00
|
|
|
|
2018-09-26 17:09:36 +00:00
|
|
|
from gn_helpers import ToGNString
|
2016-07-01 18:09:28 +00:00
|
|
|
|
2021-04-15 10:10:20 +00:00
|
|
|
# VS 2019 16.61 with 10.0.19041 SDK, and 10.0.17134 version of
|
|
|
|
# d3dcompiler_47.dll, with ARM64 libraries and UWP support.
|
|
|
|
# See go/chromium-msvc-toolchain for instructions about how to update the
|
|
|
|
# toolchain.
|
|
|
|
#
|
|
|
|
# When updating the toolchain, consider the following areas impacted by the
|
|
|
|
# toolchain version:
|
|
|
|
#
|
|
|
|
# * //base/win/windows_version.cc NTDDI preprocessor check
|
|
|
|
# Triggers a compiler error if the available SDK is older than the minimum.
|
|
|
|
# * //build/config/win/BUILD.gn NTDDI_VERSION value
|
|
|
|
# Affects the availability of APIs in the toolchain headers.
|
|
|
|
# * //docs/windows_build_instructions.md mentions of VS or Windows SDK.
|
|
|
|
# Keeps the document consistent with the toolchain version.
|
|
|
|
TOOLCHAIN_HASH = '20d5f2553f'
|
|
|
|
|
2016-07-01 18:09:28 +00:00
|
|
|
script_dir = os.path.dirname(os.path.realpath(__file__))
|
2021-04-15 10:10:20 +00:00
|
|
|
dart_src = os.path.abspath(os.path.join(script_dir, os.pardir))
|
|
|
|
sys.path.insert(0, os.path.join(dart_src, 'tools'))
|
2016-07-01 18:09:28 +00:00
|
|
|
json_data_file = os.path.join(script_dir, 'win_toolchain.json')
|
|
|
|
|
2019-07-31 20:01:01 +00:00
|
|
|
# VS versions are listed in descending order of priority (highest first).
|
|
|
|
MSVS_VERSIONS = collections.OrderedDict([
|
2019-08-05 20:34:31 +00:00
|
|
|
('2019', '16.0'),
|
2021-04-15 10:10:20 +00:00
|
|
|
('2017', '15.0'),
|
2019-07-31 20:01:01 +00:00
|
|
|
])
|
2016-10-04 16:19:07 +00:00
|
|
|
|
2021-04-15 10:10:20 +00:00
|
|
|
# List of preferred VC toolset version based on MSVS
|
|
|
|
MSVC_TOOLSET_VERSION = {
|
|
|
|
'2019': 'VC142',
|
|
|
|
'2017': 'VC141',
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
def _HostIsWindows():
|
|
|
|
"""Returns True if running on a Windows host (including under cygwin)."""
|
|
|
|
return sys.platform in ('win32', 'cygwin')
|
|
|
|
|
2016-10-04 16:19:07 +00:00
|
|
|
|
2016-07-01 18:09:28 +00:00
|
|
|
def SetEnvironmentAndGetRuntimeDllDirs():
|
2019-08-05 20:34:31 +00:00
|
|
|
"""Sets up os.environ to use the depot_tools VS toolchain with gyp, and
|
2019-07-31 20:01:01 +00:00
|
|
|
returns the location of the VC runtime DLLs so they can be copied into
|
2016-07-01 18:09:28 +00:00
|
|
|
the output directory after gyp generation.
|
2016-10-04 16:19:07 +00:00
|
|
|
|
2019-07-31 20:01:01 +00:00
|
|
|
Return value is [x64path, x86path, 'Arm64Unused'] or None. arm64path is
|
|
|
|
generated separately because there are multiple folders for the arm64 VC
|
|
|
|
runtime.
|
2016-07-01 18:09:28 +00:00
|
|
|
"""
|
2019-08-05 20:34:31 +00:00
|
|
|
vs_runtime_dll_dirs = None
|
|
|
|
depot_tools_win_toolchain = \
|
|
|
|
bool(int(os.environ.get('DEPOT_TOOLS_WIN_TOOLCHAIN', '1')))
|
|
|
|
# When running on a non-Windows host, only do this if the SDK has explicitly
|
|
|
|
# been downloaded before (in which case json_data_file will exist).
|
2021-04-15 10:10:20 +00:00
|
|
|
if ((_HostIsWindows() or os.path.exists(json_data_file)) and
|
|
|
|
depot_tools_win_toolchain):
|
2019-08-05 20:34:31 +00:00
|
|
|
if ShouldUpdateToolchain():
|
|
|
|
if len(sys.argv) > 1 and sys.argv[1] == 'update':
|
|
|
|
update_result = Update()
|
|
|
|
else:
|
|
|
|
update_result = Update(no_download=True)
|
|
|
|
if update_result != 0:
|
2021-04-15 10:10:20 +00:00
|
|
|
raise Exception('Failed to update, error code %d.' %
|
|
|
|
update_result)
|
2019-08-05 20:34:31 +00:00
|
|
|
with open(json_data_file, 'r') as tempf:
|
|
|
|
toolchain_data = json.load(tempf)
|
|
|
|
|
|
|
|
toolchain = toolchain_data['path']
|
|
|
|
version = toolchain_data['version']
|
|
|
|
win_sdk = toolchain_data.get('win_sdk')
|
|
|
|
wdk = toolchain_data['wdk']
|
|
|
|
# TODO(scottmg): The order unfortunately matters in these. They should be
|
|
|
|
# split into separate keys for x64/x86/arm64. (See CopyDlls call below).
|
|
|
|
# http://crbug.com/345992
|
|
|
|
vs_runtime_dll_dirs = toolchain_data['runtime_dirs']
|
|
|
|
# The number of runtime_dirs in the toolchain_data was two (x64/x86) but
|
|
|
|
# changed to three (x64/x86/arm64) and this code needs to handle both
|
|
|
|
# possibilities, which can change independently from this code.
|
|
|
|
if len(vs_runtime_dll_dirs) == 2:
|
|
|
|
vs_runtime_dll_dirs.append('Arm64Unused')
|
|
|
|
|
|
|
|
os.environ['GYP_MSVS_OVERRIDE_PATH'] = toolchain
|
|
|
|
|
|
|
|
os.environ['WINDOWSSDKDIR'] = win_sdk
|
|
|
|
os.environ['WDK_DIR'] = wdk
|
|
|
|
# Include the VS runtime in the PATH in case it's not machine-installed.
|
|
|
|
runtime_path = os.path.pathsep.join(vs_runtime_dll_dirs)
|
|
|
|
os.environ['PATH'] = runtime_path + os.path.pathsep + os.environ['PATH']
|
|
|
|
elif sys.platform == 'win32' and not depot_tools_win_toolchain:
|
|
|
|
if not 'GYP_MSVS_OVERRIDE_PATH' in os.environ:
|
|
|
|
os.environ['GYP_MSVS_OVERRIDE_PATH'] = DetectVisualStudioPath()
|
|
|
|
|
|
|
|
# When using an installed toolchain these files aren't needed in the output
|
|
|
|
# directory in order to run binaries locally, but they are needed in order
|
|
|
|
# to create isolates or the mini_installer. Copying them to the output
|
|
|
|
# directory ensures that they are available when needed.
|
|
|
|
bitness = platform.architecture()[0]
|
|
|
|
# When running 64-bit python the x64 DLLs will be in System32
|
|
|
|
# ARM64 binaries will not be available in the system directories because we
|
|
|
|
# don't build on ARM64 machines.
|
|
|
|
x64_path = 'System32' if bitness == '64bit' else 'Sysnative'
|
|
|
|
x64_path = os.path.join(os.path.expandvars('%windir%'), x64_path)
|
|
|
|
vs_runtime_dll_dirs = [
|
|
|
|
x64_path,
|
|
|
|
os.path.join(os.path.expandvars('%windir%'), 'SysWOW64'),
|
|
|
|
'Arm64Unused'
|
|
|
|
]
|
|
|
|
|
|
|
|
return vs_runtime_dll_dirs
|
2016-10-04 16:19:07 +00:00
|
|
|
|
|
|
|
|
|
|
|
def _RegistryGetValueUsingWinReg(key, value):
|
2019-08-05 20:34:31 +00:00
|
|
|
"""Use the _winreg module to obtain the value of a registry key.
|
2016-10-04 16:19:07 +00:00
|
|
|
|
|
|
|
Args:
|
|
|
|
key: The registry key.
|
|
|
|
value: The particular registry value to read.
|
|
|
|
Return:
|
|
|
|
contents of the registry key's value, or None on failure. Throws
|
|
|
|
ImportError if _winreg is unavailable.
|
|
|
|
"""
|
2019-08-05 20:34:31 +00:00
|
|
|
import _winreg
|
|
|
|
try:
|
|
|
|
root, subkey = key.split('\\', 1)
|
|
|
|
assert root == 'HKLM' # Only need HKLM for now.
|
|
|
|
with _winreg.OpenKey(_winreg.HKEY_LOCAL_MACHINE, subkey) as hkey:
|
|
|
|
return _winreg.QueryValueEx(hkey, value)[0]
|
|
|
|
except WindowsError:
|
|
|
|
return None
|
2016-10-04 16:19:07 +00:00
|
|
|
|
|
|
|
|
|
|
|
def _RegistryGetValue(key, value):
|
2019-08-05 20:34:31 +00:00
|
|
|
try:
|
|
|
|
return _RegistryGetValueUsingWinReg(key, value)
|
|
|
|
except ImportError:
|
|
|
|
raise Exception('The python library _winreg not found.')
|
2016-10-04 16:19:07 +00:00
|
|
|
|
|
|
|
|
|
|
|
def GetVisualStudioVersion():
|
2019-08-05 20:34:31 +00:00
|
|
|
"""Return best available version of Visual Studio.
|
2016-10-04 16:19:07 +00:00
|
|
|
"""
|
2021-04-15 10:10:20 +00:00
|
|
|
supported_versions = list(MSVS_VERSIONS.keys())
|
2019-07-31 20:01:01 +00:00
|
|
|
|
2019-08-05 20:34:31 +00:00
|
|
|
# VS installed in depot_tools for Googlers
|
|
|
|
if bool(int(os.environ.get('DEPOT_TOOLS_WIN_TOOLCHAIN', '1'))):
|
|
|
|
return supported_versions[0]
|
2019-07-31 20:01:01 +00:00
|
|
|
|
2019-08-05 20:34:31 +00:00
|
|
|
# VS installed in system for external developers
|
|
|
|
supported_versions_str = ', '.join(
|
|
|
|
'{} ({})'.format(v, k) for k, v in MSVS_VERSIONS.items())
|
|
|
|
available_versions = []
|
|
|
|
for version in supported_versions:
|
2021-04-15 10:10:20 +00:00
|
|
|
# Checking vs%s_install environment variables.
|
|
|
|
# For example, vs2019_install could have the value
|
|
|
|
# "C:\Program Files (x86)\Microsoft Visual Studio\2019\Community".
|
|
|
|
# Only vs2017_install and vs2019_install are supported.
|
|
|
|
path = os.environ.get('vs%s_install' % version)
|
|
|
|
if path and os.path.exists(path):
|
|
|
|
available_versions.append(version)
|
|
|
|
break
|
|
|
|
# Detecting VS under possible paths.
|
|
|
|
path = os.path.expandvars('%ProgramFiles(x86)%' +
|
|
|
|
'/Microsoft Visual Studio/%s' % version)
|
|
|
|
if path and any(
|
|
|
|
os.path.exists(os.path.join(path, edition))
|
|
|
|
for edition in ('Enterprise', 'Professional', 'Community',
|
|
|
|
'Preview', 'BuildTools')):
|
|
|
|
available_versions.append(version)
|
|
|
|
break
|
2019-07-31 20:01:01 +00:00
|
|
|
|
2019-08-05 20:34:31 +00:00
|
|
|
if not available_versions:
|
|
|
|
raise Exception('No supported Visual Studio can be found.'
|
|
|
|
' Supported versions are: %s.' % supported_versions_str)
|
|
|
|
return available_versions[0]
|
2016-10-04 16:19:07 +00:00
|
|
|
|
|
|
|
|
|
|
|
def DetectVisualStudioPath():
|
2021-04-15 10:10:20 +00:00
|
|
|
"""Return path to the installed Visual Studio.
|
2016-10-04 16:19:07 +00:00
|
|
|
"""
|
|
|
|
|
2019-08-05 20:34:31 +00:00
|
|
|
# Note that this code is used from
|
|
|
|
# build/toolchain/win/setup_toolchain.py as well.
|
|
|
|
version_as_year = GetVisualStudioVersion()
|
|
|
|
|
|
|
|
# The VC++ >=2017 install location needs to be located using COM instead of
|
|
|
|
# the registry. For details see:
|
|
|
|
# https://blogs.msdn.microsoft.com/heaths/2016/09/15/changes-to-visual-studio-15-setup/
|
|
|
|
# For now we use a hardcoded default with an environment variable override.
|
2021-04-15 10:10:20 +00:00
|
|
|
for path in (os.environ.get('vs%s_install' % version_as_year),
|
|
|
|
os.path.expandvars('%ProgramFiles(x86)%' +
|
|
|
|
'/Microsoft Visual Studio/%s/Enterprise' %
|
|
|
|
version_as_year),
|
|
|
|
os.path.expandvars('%ProgramFiles(x86)%' +
|
|
|
|
'/Microsoft Visual Studio/%s/Professional' %
|
|
|
|
version_as_year),
|
|
|
|
os.path.expandvars('%ProgramFiles(x86)%' +
|
|
|
|
'/Microsoft Visual Studio/%s/Community' %
|
|
|
|
version_as_year),
|
|
|
|
os.path.expandvars('%ProgramFiles(x86)%' +
|
|
|
|
'/Microsoft Visual Studio/%s/Preview' %
|
|
|
|
version_as_year),
|
|
|
|
os.path.expandvars('%ProgramFiles(x86)%' +
|
|
|
|
'/Microsoft Visual Studio/%s/BuildTools' %
|
|
|
|
version_as_year)):
|
2019-08-05 20:34:31 +00:00
|
|
|
if path and os.path.exists(path):
|
|
|
|
return path
|
|
|
|
|
2021-04-15 10:10:20 +00:00
|
|
|
raise Exception('Visual Studio Version %s not found.' % version_as_year)
|
2016-07-01 18:09:28 +00:00
|
|
|
|
|
|
|
|
2016-10-04 16:19:07 +00:00
|
|
|
def _CopyRuntimeImpl(target, source, verbose=True):
|
2019-08-05 20:34:31 +00:00
|
|
|
"""Copy |source| to |target| if it doesn't already exist or if it needs to be
|
2016-10-04 16:19:07 +00:00
|
|
|
updated (comparing last modified time as an approximate float match as for
|
|
|
|
some reason the values tend to differ by ~1e-07 despite being copies of the
|
|
|
|
same file... https://crbug.com/603603).
|
2016-07-01 18:09:28 +00:00
|
|
|
"""
|
2019-08-05 20:34:31 +00:00
|
|
|
if (os.path.isdir(os.path.dirname(target)) and
|
|
|
|
(not os.path.isfile(target) or
|
|
|
|
abs(os.stat(target).st_mtime - os.stat(source).st_mtime) >= 0.01)):
|
|
|
|
if verbose:
|
|
|
|
print('Copying %s to %s...' % (source, target))
|
|
|
|
if os.path.exists(target):
|
|
|
|
# Make the file writable so that we can delete it now, and keep it
|
|
|
|
# readable.
|
|
|
|
os.chmod(target, stat.S_IWRITE | stat.S_IREAD)
|
|
|
|
os.unlink(target)
|
|
|
|
shutil.copy2(source, target)
|
|
|
|
# Make the file writable so that we can overwrite or delete it later,
|
|
|
|
# keep it readable.
|
|
|
|
os.chmod(target, stat.S_IWRITE | stat.S_IREAD)
|
|
|
|
|
2016-07-01 18:09:28 +00:00
|
|
|
|
2019-07-31 20:01:01 +00:00
|
|
|
def _SortByHighestVersionNumberFirst(list_of_str_versions):
|
2019-08-05 20:34:31 +00:00
|
|
|
"""This sorts |list_of_str_versions| according to version number rules
|
2019-07-31 20:01:01 +00:00
|
|
|
so that version "1.12" is higher than version "1.9". Does not work
|
|
|
|
with non-numeric versions like 1.4.a8 which will be higher than
|
|
|
|
1.4.a12. It does handle the versions being embedded in file paths.
|
|
|
|
"""
|
|
|
|
|
2019-08-05 20:34:31 +00:00
|
|
|
def to_int_if_int(x):
|
|
|
|
try:
|
|
|
|
return int(x)
|
|
|
|
except ValueError:
|
|
|
|
return x
|
|
|
|
|
|
|
|
def to_number_sequence(x):
|
|
|
|
part_sequence = re.split(r'[\\/\.]', x)
|
|
|
|
return [to_int_if_int(x) for x in part_sequence]
|
|
|
|
|
|
|
|
list_of_str_versions.sort(key=to_number_sequence, reverse=True)
|
2019-07-31 20:01:01 +00:00
|
|
|
|
2016-07-01 18:09:28 +00:00
|
|
|
|
2021-04-15 10:10:20 +00:00
|
|
|
def _CopyUCRTRuntime(target_dir, source_dir, target_cpu, suffix):
|
2019-08-05 20:34:31 +00:00
|
|
|
"""Copy both the msvcp and vccorlib runtime DLLs, only if the target doesn't
|
2016-07-01 18:09:28 +00:00
|
|
|
exist, but the target directory does exist."""
|
2019-08-05 20:34:31 +00:00
|
|
|
if target_cpu == 'arm64':
|
|
|
|
# Windows ARM64 VCRuntime is located at {toolchain_root}/VC/Redist/MSVC/
|
2021-04-15 10:10:20 +00:00
|
|
|
# {x.y.z}/[debug_nonredist/]arm64/Microsoft.VC14x.CRT/.
|
|
|
|
# Select VC toolset directory based on Visual Studio version
|
2019-08-05 20:34:31 +00:00
|
|
|
vc_redist_root = FindVCRedistRoot()
|
|
|
|
if suffix.startswith('.'):
|
2021-04-15 10:10:20 +00:00
|
|
|
vc_toolset_dir = 'Microsoft.{}.CRT' \
|
|
|
|
.format(MSVC_TOOLSET_VERSION[GetVisualStudioVersion()])
|
|
|
|
source_dir = os.path.join(vc_redist_root, 'arm64', vc_toolset_dir)
|
2019-08-05 20:34:31 +00:00
|
|
|
else:
|
2021-04-15 10:10:20 +00:00
|
|
|
vc_toolset_dir = 'Microsoft.{}.DebugCRT' \
|
|
|
|
.format(MSVC_TOOLSET_VERSION[GetVisualStudioVersion()])
|
2019-08-05 20:34:31 +00:00
|
|
|
source_dir = os.path.join(vc_redist_root, 'debug_nonredist',
|
2021-04-15 10:10:20 +00:00
|
|
|
'arm64', vc_toolset_dir)
|
|
|
|
file_parts = ('msvcp140', 'vccorlib140', 'vcruntime140')
|
|
|
|
if target_cpu == 'x64' and GetVisualStudioVersion() != '2017':
|
|
|
|
file_parts = file_parts + ('vcruntime140_1',)
|
|
|
|
for file_part in file_parts:
|
|
|
|
dll = file_part + suffix
|
2019-08-05 20:34:31 +00:00
|
|
|
target = os.path.join(target_dir, dll)
|
|
|
|
source = os.path.join(source_dir, dll)
|
|
|
|
_CopyRuntimeImpl(target, source)
|
|
|
|
# Copy the UCRT files from the Windows SDK. This location includes the
|
|
|
|
# api-ms-win-crt-*.dll files that are not found in the Windows directory.
|
|
|
|
# These files are needed for component builds. If WINDOWSSDKDIR is not set
|
|
|
|
# use the default SDK path. This will be the case when
|
|
|
|
# DEPOT_TOOLS_WIN_TOOLCHAIN=0 and vcvarsall.bat has not been run.
|
|
|
|
win_sdk_dir = os.path.normpath(
|
|
|
|
os.environ.get(
|
|
|
|
'WINDOWSSDKDIR',
|
|
|
|
os.path.expandvars('%ProgramFiles(x86)%'
|
|
|
|
'\\Windows Kits\\10')))
|
|
|
|
# ARM64 doesn't have a redist for the ucrt DLLs because they are always
|
|
|
|
# present in the OS.
|
|
|
|
if target_cpu != 'arm64':
|
|
|
|
# Starting with the 10.0.17763 SDK the ucrt files are in a version-named
|
|
|
|
# directory - this handles both cases.
|
|
|
|
redist_dir = os.path.join(win_sdk_dir, 'Redist')
|
|
|
|
version_dirs = glob.glob(os.path.join(redist_dir, '10.*'))
|
|
|
|
if len(version_dirs) > 0:
|
|
|
|
_SortByHighestVersionNumberFirst(version_dirs)
|
|
|
|
redist_dir = version_dirs[0]
|
|
|
|
ucrt_dll_dirs = os.path.join(redist_dir, 'ucrt', 'DLLs', target_cpu)
|
|
|
|
ucrt_files = glob.glob(os.path.join(ucrt_dll_dirs, 'api-ms-win-*.dll'))
|
|
|
|
assert len(ucrt_files) > 0
|
|
|
|
for ucrt_src_file in ucrt_files:
|
|
|
|
file_part = os.path.basename(ucrt_src_file)
|
|
|
|
ucrt_dst_file = os.path.join(target_dir, file_part)
|
|
|
|
_CopyRuntimeImpl(ucrt_dst_file, ucrt_src_file, False)
|
|
|
|
# We must copy ucrtbase.dll for x64/x86, and ucrtbased.dll for all CPU types.
|
|
|
|
if target_cpu != 'arm64' or not suffix.startswith('.'):
|
|
|
|
if not suffix.startswith('.'):
|
|
|
|
# ucrtbased.dll is located at {win_sdk_dir}/bin/{a.b.c.d}/{target_cpu}/
|
|
|
|
# ucrt/.
|
2021-04-15 10:10:20 +00:00
|
|
|
sdk_bin_root = os.path.join(win_sdk_dir, 'bin')
|
|
|
|
sdk_bin_sub_dirs = glob.glob(os.path.join(sdk_bin_root, '10.*'))
|
2019-08-05 20:34:31 +00:00
|
|
|
# Select the most recent SDK if there are multiple versions installed.
|
|
|
|
_SortByHighestVersionNumberFirst(sdk_bin_sub_dirs)
|
|
|
|
for directory in sdk_bin_sub_dirs:
|
2021-04-15 10:10:20 +00:00
|
|
|
sdk_redist_root_version = os.path.join(sdk_bin_root, directory)
|
2019-08-05 20:34:31 +00:00
|
|
|
if not os.path.isdir(sdk_redist_root_version):
|
|
|
|
continue
|
2021-04-15 10:10:20 +00:00
|
|
|
source_dir = os.path.join(sdk_redist_root_version, target_cpu,
|
|
|
|
'ucrt')
|
|
|
|
break
|
|
|
|
_CopyRuntimeImpl(os.path.join(target_dir, 'ucrtbase' + suffix),
|
|
|
|
os.path.join(source_dir, 'ucrtbase' + suffix))
|
2016-10-04 16:19:07 +00:00
|
|
|
|
|
|
|
|
2019-07-31 20:01:01 +00:00
|
|
|
def FindVCComponentRoot(component):
|
2019-08-05 20:34:31 +00:00
|
|
|
"""Find the most recent Tools or Redist or other directory in an MSVC install.
|
2019-07-31 20:01:01 +00:00
|
|
|
Typical results are {toolchain_root}/VC/{component}/MSVC/{x.y.z}. The {x.y.z}
|
|
|
|
version number part changes frequently so the highest version number found is
|
|
|
|
used.
|
2017-09-28 09:25:37 +00:00
|
|
|
"""
|
2019-07-31 20:01:01 +00:00
|
|
|
|
2019-08-05 20:34:31 +00:00
|
|
|
SetEnvironmentAndGetRuntimeDllDirs()
|
|
|
|
assert ('GYP_MSVS_OVERRIDE_PATH' in os.environ)
|
|
|
|
vc_component_msvc_root = os.path.join(os.environ['GYP_MSVS_OVERRIDE_PATH'],
|
|
|
|
'VC', component, 'MSVC')
|
2021-04-15 10:10:20 +00:00
|
|
|
vc_component_msvc_contents = glob.glob(
|
|
|
|
os.path.join(vc_component_msvc_root, '14.*'))
|
2019-08-05 20:34:31 +00:00
|
|
|
# Select the most recent toolchain if there are several.
|
|
|
|
_SortByHighestVersionNumberFirst(vc_component_msvc_contents)
|
|
|
|
for directory in vc_component_msvc_contents:
|
2021-04-15 10:10:20 +00:00
|
|
|
if os.path.isdir(directory):
|
|
|
|
return directory
|
2019-08-05 20:34:31 +00:00
|
|
|
raise Exception('Unable to find the VC %s directory.' % component)
|
2019-07-31 20:01:01 +00:00
|
|
|
|
2017-09-28 09:25:37 +00:00
|
|
|
|
2019-07-31 20:01:01 +00:00
|
|
|
def FindVCRedistRoot():
|
2019-08-05 20:34:31 +00:00
|
|
|
"""In >=VS2017, Redist binaries are located in
|
2019-07-31 20:01:01 +00:00
|
|
|
{toolchain_root}/VC/Redist/MSVC/{x.y.z}/{target_cpu}/.
|
2016-10-04 16:19:07 +00:00
|
|
|
|
2019-07-31 20:01:01 +00:00
|
|
|
This returns the '{toolchain_root}/VC/Redist/MSVC/{x.y.z}/' path.
|
2017-09-28 09:25:37 +00:00
|
|
|
"""
|
2019-08-05 20:34:31 +00:00
|
|
|
return FindVCComponentRoot('Redist')
|
2017-09-27 09:40:38 +00:00
|
|
|
|
|
|
|
|
2017-09-28 09:25:37 +00:00
|
|
|
def _CopyRuntime(target_dir, source_dir, target_cpu, debug):
|
2019-08-05 20:34:31 +00:00
|
|
|
"""Copy the VS runtime DLLs, only if the target doesn't exist, but the target
|
2019-07-31 20:01:01 +00:00
|
|
|
directory does exist. Handles VS 2015, 2017 and 2019."""
|
2019-08-05 20:34:31 +00:00
|
|
|
suffix = 'd.dll' if debug else '.dll'
|
|
|
|
# VS 2015, 2017 and 2019 use the same CRT DLLs.
|
2021-04-15 10:10:20 +00:00
|
|
|
_CopyUCRTRuntime(target_dir, source_dir, target_cpu, suffix)
|
2016-07-01 18:09:28 +00:00
|
|
|
|
|
|
|
|
|
|
|
def CopyDlls(target_dir, configuration, target_cpu):
|
2019-08-05 20:34:31 +00:00
|
|
|
"""Copy the VS runtime DLLs into the requested directory as needed.
|
2016-07-01 18:09:28 +00:00
|
|
|
|
|
|
|
configuration is one of 'Debug' or 'Release'.
|
2019-07-31 20:01:01 +00:00
|
|
|
target_cpu is one of 'x86', 'x64' or 'arm64'.
|
2016-07-01 18:09:28 +00:00
|
|
|
|
|
|
|
The debug configuration gets both the debug and release DLLs; the
|
|
|
|
release config only the latter.
|
|
|
|
"""
|
2019-08-05 20:34:31 +00:00
|
|
|
vs_runtime_dll_dirs = SetEnvironmentAndGetRuntimeDllDirs()
|
|
|
|
if not vs_runtime_dll_dirs:
|
|
|
|
return
|
|
|
|
|
|
|
|
x64_runtime, x86_runtime, arm64_runtime = vs_runtime_dll_dirs
|
|
|
|
if target_cpu == 'x64':
|
|
|
|
runtime_dir = x64_runtime
|
|
|
|
elif target_cpu == 'x86':
|
|
|
|
runtime_dir = x86_runtime
|
|
|
|
elif target_cpu == 'arm64':
|
|
|
|
runtime_dir = arm64_runtime
|
|
|
|
else:
|
|
|
|
raise Exception('Unknown target_cpu: ' + target_cpu)
|
|
|
|
_CopyRuntime(target_dir, runtime_dir, target_cpu, debug=False)
|
|
|
|
if configuration == 'Debug':
|
|
|
|
_CopyRuntime(target_dir, runtime_dir, target_cpu, debug=True)
|
|
|
|
_CopyDebugger(target_dir, target_cpu)
|
2017-09-28 09:25:37 +00:00
|
|
|
|
|
|
|
|
|
|
|
def _CopyDebugger(target_dir, target_cpu):
|
2019-08-05 20:34:31 +00:00
|
|
|
"""Copy dbghelp.dll and dbgcore.dll into the requested directory as needed.
|
2017-09-28 09:25:37 +00:00
|
|
|
|
2019-07-31 20:01:01 +00:00
|
|
|
target_cpu is one of 'x86', 'x64' or 'arm64'.
|
2017-09-28 09:25:37 +00:00
|
|
|
|
|
|
|
dbghelp.dll is used when Chrome needs to symbolize stacks. Copying this file
|
|
|
|
from the SDK directory avoids using the system copy of dbghelp.dll which then
|
|
|
|
ensures compatibility with recent debug information formats, such as VS
|
|
|
|
2017 /debug:fastlink PDBs.
|
|
|
|
|
|
|
|
dbgcore.dll is needed when using some functions from dbghelp.dll (like
|
|
|
|
MinidumpWriteDump).
|
|
|
|
"""
|
2019-08-05 20:34:31 +00:00
|
|
|
win_sdk_dir = SetEnvironmentAndGetSDKDir()
|
|
|
|
if not win_sdk_dir:
|
|
|
|
return
|
|
|
|
|
|
|
|
# List of debug files that should be copied, the first element of the tuple is
|
|
|
|
# the name of the file and the second indicates if it's optional.
|
|
|
|
debug_files = [('dbghelp.dll', False), ('dbgcore.dll', True)]
|
2021-04-15 10:10:20 +00:00
|
|
|
# The UCRT is not a redistributable component on arm64.
|
|
|
|
if target_cpu != 'arm64':
|
|
|
|
debug_files.extend([('api-ms-win-downlevel-kernel32-l2-1-0.dll', False),
|
|
|
|
('api-ms-win-eventing-provider-l1-1-0.dll', False)])
|
2019-08-05 20:34:31 +00:00
|
|
|
for debug_file, is_optional in debug_files:
|
|
|
|
full_path = os.path.join(win_sdk_dir, 'Debuggers', target_cpu,
|
|
|
|
debug_file)
|
|
|
|
if not os.path.exists(full_path):
|
|
|
|
if is_optional:
|
|
|
|
continue
|
|
|
|
else:
|
|
|
|
raise Exception(
|
2021-04-15 10:10:20 +00:00
|
|
|
'%s not found in "%s"\r\nYou must install'
|
|
|
|
'Windows 10 SDK version 10.0.19041.0 including the '
|
|
|
|
'"Debugging Tools for Windows" feature.' %
|
|
|
|
(debug_file, full_path))
|
2019-08-05 20:34:31 +00:00
|
|
|
target_path = os.path.join(target_dir, debug_file)
|
|
|
|
_CopyRuntimeImpl(target_path, full_path)
|
2016-07-01 18:09:28 +00:00
|
|
|
|
|
|
|
|
|
|
|
def _GetDesiredVsToolchainHashes():
|
2019-08-05 20:34:31 +00:00
|
|
|
"""Load a list of SHA1s corresponding to the toolchains that we want installed
|
2016-07-01 18:09:28 +00:00
|
|
|
to build with."""
|
2021-04-15 10:10:20 +00:00
|
|
|
# Third parties that do not have access to the canonical toolchain can map
|
|
|
|
# canonical toolchain version to their own toolchain versions.
|
|
|
|
toolchain_hash_mapping_key = 'GYP_MSVS_HASH_%s' % TOOLCHAIN_HASH
|
|
|
|
return [os.environ.get(toolchain_hash_mapping_key, TOOLCHAIN_HASH)]
|
2016-10-04 16:19:07 +00:00
|
|
|
|
|
|
|
|
|
|
|
def ShouldUpdateToolchain():
|
2019-08-05 20:34:31 +00:00
|
|
|
"""Check if the toolchain should be upgraded."""
|
|
|
|
if not os.path.exists(json_data_file):
|
|
|
|
return True
|
|
|
|
with open(json_data_file, 'r') as tempf:
|
|
|
|
toolchain_data = json.load(tempf)
|
|
|
|
version = toolchain_data['version']
|
|
|
|
env_version = GetVisualStudioVersion()
|
|
|
|
# If there's a mismatch between the version set in the environment and the one
|
|
|
|
# in the json file then the toolchain should be updated.
|
|
|
|
return version != env_version
|
2016-07-01 18:09:28 +00:00
|
|
|
|
|
|
|
|
2019-07-31 20:01:01 +00:00
|
|
|
def Update(force=False, no_download=False):
|
2019-08-05 20:34:31 +00:00
|
|
|
"""Requests an update of the toolchain to the specific hashes we have at
|
2016-07-01 18:09:28 +00:00
|
|
|
this revision. The update outputs a .json of the various configuration
|
|
|
|
information required to pass to gyp which we use in |GetToolchainDir()|.
|
2019-07-31 20:01:01 +00:00
|
|
|
If no_download is true then the toolchain will be configured if present but
|
|
|
|
will not be downloaded.
|
2016-07-01 18:09:28 +00:00
|
|
|
"""
|
2019-08-05 20:34:31 +00:00
|
|
|
if force != False and force != '--force':
|
|
|
|
print('Unknown parameter "%s"' % force, file=sys.stderr)
|
|
|
|
return 1
|
|
|
|
if force == '--force' or os.path.exists(json_data_file):
|
|
|
|
force = True
|
|
|
|
|
|
|
|
depot_tools_win_toolchain = \
|
|
|
|
bool(int(os.environ.get('DEPOT_TOOLS_WIN_TOOLCHAIN', '1')))
|
2021-04-15 10:10:20 +00:00
|
|
|
if (_HostIsWindows() or force) and depot_tools_win_toolchain:
|
2019-08-05 20:34:31 +00:00
|
|
|
import find_depot_tools
|
|
|
|
depot_tools_path = find_depot_tools.add_depot_tools_to_path()
|
|
|
|
|
|
|
|
# On Linux, the file system is usually case-sensitive while the Windows
|
|
|
|
# SDK only works on case-insensitive file systems. If it doesn't already
|
|
|
|
# exist, set up a ciopfs fuse mount to put the SDK in a case-insensitive
|
|
|
|
# part of the file system.
|
|
|
|
toolchain_dir = os.path.join(depot_tools_path, 'win_toolchain',
|
|
|
|
'vs_files')
|
|
|
|
# For testing this block, unmount existing mounts with
|
|
|
|
# fusermount -u third_party/depot_tools/win_toolchain/vs_files
|
|
|
|
if sys.platform.startswith(
|
|
|
|
'linux') and not os.path.ismount(toolchain_dir):
|
|
|
|
import distutils.spawn
|
|
|
|
ciopfs = distutils.spawn.find_executable('ciopfs')
|
|
|
|
if not ciopfs:
|
|
|
|
# ciopfs not found in PATH; try the one downloaded from the DEPS hook.
|
|
|
|
ciopfs = os.path.join(script_dir, 'ciopfs')
|
|
|
|
if not os.path.isdir(toolchain_dir):
|
|
|
|
os.mkdir(toolchain_dir)
|
|
|
|
if not os.path.isdir(toolchain_dir + '.ciopfs'):
|
|
|
|
os.mkdir(toolchain_dir + '.ciopfs')
|
|
|
|
# Without use_ino, clang's #pragma once and Wnonportable-include-path
|
|
|
|
# both don't work right, see https://llvm.org/PR34931
|
|
|
|
# use_ino doesn't slow down builds, so it seems there's no drawback to
|
|
|
|
# just using it always.
|
|
|
|
subprocess.check_call([
|
|
|
|
ciopfs, '-o', 'use_ino', toolchain_dir + '.ciopfs',
|
|
|
|
toolchain_dir
|
|
|
|
])
|
|
|
|
|
|
|
|
get_toolchain_args = [
|
2021-04-15 10:10:20 +00:00
|
|
|
# TODO(athom): use sys.executable (python3).
|
2021-05-21 11:04:22 +00:00
|
|
|
# Note: depot_tools contains python.bat not python.exe
|
|
|
|
# so for python to land on the first python in the PATH
|
|
|
|
# irrespective of its extension we pass shell=True below.
|
2021-04-15 10:10:20 +00:00
|
|
|
'python',
|
2019-08-05 20:34:31 +00:00
|
|
|
os.path.join(depot_tools_path, 'win_toolchain',
|
|
|
|
'get_toolchain_if_necessary.py'),
|
|
|
|
'--output-json',
|
|
|
|
json_data_file,
|
|
|
|
] + _GetDesiredVsToolchainHashes()
|
|
|
|
if force:
|
|
|
|
get_toolchain_args.append('--force')
|
|
|
|
if no_download:
|
|
|
|
get_toolchain_args.append('--no-download')
|
2021-05-21 11:04:22 +00:00
|
|
|
subprocess.check_call(get_toolchain_args, shell=True)
|
2019-08-05 20:34:31 +00:00
|
|
|
|
|
|
|
return 0
|
2016-07-01 18:09:28 +00:00
|
|
|
|
|
|
|
|
2016-10-04 16:19:07 +00:00
|
|
|
def NormalizePath(path):
|
2019-08-05 20:34:31 +00:00
|
|
|
while path.endswith('\\'):
|
|
|
|
path = path[:-1]
|
|
|
|
return path
|
2016-10-04 16:19:07 +00:00
|
|
|
|
|
|
|
|
2017-09-28 09:25:37 +00:00
|
|
|
def SetEnvironmentAndGetSDKDir():
|
2019-08-05 20:34:31 +00:00
|
|
|
"""Gets location information about the current sdk (must have been
|
2016-07-01 18:09:28 +00:00
|
|
|
previously updated by 'update'). This is used for the GN build."""
|
2019-08-05 20:34:31 +00:00
|
|
|
SetEnvironmentAndGetRuntimeDllDirs()
|
2016-07-01 18:09:28 +00:00
|
|
|
|
2019-08-05 20:34:31 +00:00
|
|
|
# If WINDOWSSDKDIR is not set, search the default SDK path and set it.
|
|
|
|
if not 'WINDOWSSDKDIR' in os.environ:
|
|
|
|
default_sdk_path = os.path.expandvars('%ProgramFiles(x86)%'
|
|
|
|
'\\Windows Kits\\10')
|
|
|
|
if os.path.isdir(default_sdk_path):
|
|
|
|
os.environ['WINDOWSSDKDIR'] = default_sdk_path
|
2016-07-01 18:09:28 +00:00
|
|
|
|
2019-08-05 20:34:31 +00:00
|
|
|
return NormalizePath(os.environ['WINDOWSSDKDIR'])
|
2017-09-28 09:25:37 +00:00
|
|
|
|
|
|
|
|
|
|
|
def GetToolchainDir():
|
2019-08-05 20:34:31 +00:00
|
|
|
"""Gets location information about the current toolchain (must have been
|
2017-09-28 09:25:37 +00:00
|
|
|
previously updated by 'update'). This is used for the GN build."""
|
2019-08-05 20:34:31 +00:00
|
|
|
runtime_dll_dirs = SetEnvironmentAndGetRuntimeDllDirs()
|
|
|
|
win_sdk_dir = SetEnvironmentAndGetSDKDir()
|
2017-09-28 09:25:37 +00:00
|
|
|
|
2019-08-05 20:34:31 +00:00
|
|
|
print('''vs_path = %s
|
2018-09-26 17:09:36 +00:00
|
|
|
sdk_path = %s
|
|
|
|
vs_version = %s
|
|
|
|
wdk_dir = %s
|
|
|
|
runtime_dirs = %s
|
2019-07-31 20:01:01 +00:00
|
|
|
''' % (ToGNString(NormalizePath(os.environ['GYP_MSVS_OVERRIDE_PATH'])),
|
|
|
|
ToGNString(win_sdk_dir), ToGNString(GetVisualStudioVersion()),
|
|
|
|
ToGNString(NormalizePath(os.environ.get('WDK_DIR', ''))),
|
|
|
|
ToGNString(os.path.pathsep.join(runtime_dll_dirs or ['None']))))
|
2016-07-01 18:09:28 +00:00
|
|
|
|
|
|
|
|
|
|
|
def main():
|
2019-08-05 20:34:31 +00:00
|
|
|
commands = {
|
|
|
|
'update': Update,
|
|
|
|
'get_toolchain_dir': GetToolchainDir,
|
|
|
|
'copy_dlls': CopyDlls,
|
|
|
|
}
|
|
|
|
if len(sys.argv) < 2 or sys.argv[1] not in commands:
|
|
|
|
print('Expected one of: %s' % ', '.join(commands), file=sys.stderr)
|
|
|
|
return 1
|
|
|
|
return commands[sys.argv[1]](*sys.argv[2:])
|
2016-07-01 18:09:28 +00:00
|
|
|
|
|
|
|
|
|
|
|
if __name__ == '__main__':
|
2019-08-05 20:34:31 +00:00
|
|
|
sys.exit(main())
|