blob: 6b950b7bc08761031640b6456623db4ee670e602 [file] [log] [blame]
#!/usr/bin/env python
# Copyright (c) 2012 Google Inc. All rights reserved.
# Use of this source code is governed by a BSD-style license that can be
# found in the LICENSE file.
"""Utility functions for Windows builds.
These functions are executed via gyp-win-tool when using the ninja generator.
"""
import os
import shutil
import subprocess
import sys
import platform
BASE_DIR = os.path.dirname(os.path.abspath(__file__))
def main(args):
executor = WinTool()
exit_code = executor.Dispatch(args)
if exit_code is not None:
if exit_code != 0:
print("Abnormal exit (" + str(exit_code) + ") code while executing " \
+ str(args))
sys.exit(exit_code)
class WinTool(object):
"""This class performs all the Windows tooling steps. The methods can either
be executed directly, or dispatched from an argument list."""
def Dispatch(self, args):
"""Dispatches a string command to a method."""
if len(args) < 1:
raise Exception("Not enough arguments")
method = "Exec%s" % self._CommandifyName(args[0])
return getattr(self, method)(*args[1:])
def _CommandifyName(self, name_string):
"""Transforms a tool name like recursive-mirror to RecursiveMirror."""
return name_string.title().replace('-', '')
def _GetEnv(self, arch):
"""Gets the saved environment from a file for a given architecture."""
# The environment is saved as an "environment block" (see CreateProcess
# and msvs_emulation for details). We convert to a dict here.
# Drop last 2 NULs, one for list terminator, one for trailing vs. separator.
pairs = open(arch).read()[:-2].split('\0')
kvs = [item.split('=', 1) for item in pairs]
return dict(kvs)
def ExecStamp(self, path):
"""Simple stamp command."""
open(path, 'w').close()
def ExecRecursiveMirror(self, source, dest):
"""Emulation of rm -rf out && cp -af in out."""
if os.path.exists(dest):
if os.path.isdir(dest):
shutil.rmtree(dest)
else:
os.unlink(dest)
if os.path.isdir(source):
# Ignore .git directory.
shutil.copytree(source, dest, ignore=shutil.ignore_patterns(r'.git'))
else:
shutil.copy2(source, dest)
if platform.system() == 'Windows':
def ExecLinkWrapper(self, arch, *args):
"""Filter diagnostic output from link that looks like:
' Creating library ui.dll.lib and object ui.dll.exp'
This happens when there are exports from the dll or exe.
"""
env = self._GetEnv(arch)
popen = subprocess.Popen(args, shell=True, env=env,
stdout=subprocess.PIPE, stderr=subprocess.STDOUT,
universal_newlines=True)
out, _ = popen.communicate()
for line in out.splitlines():
if not line.startswith(' Creating library '):
print line
return popen.returncode
def ExecManifestWrapper(self, arch, *args):
"""Run manifest tool with environment set. Strip out undesirable warning
(some XML blocks are recognized by the OS loader, but not the manifest
tool)."""
env = self._GetEnv(arch)
popen = subprocess.Popen(args, shell=True, env=env,
stdout=subprocess.PIPE, stderr=subprocess.STDOUT,
universal_newlines=True)
out, _ = popen.communicate()
for line in out.splitlines():
if line and 'manifest authoring warning 81010002' not in line:
print line
return popen.returncode
def ExecMidlWrapper(self, arch, outdir, tlb, h, dlldata, iid, proxy, idl,
*flags):
"""Filter noisy filenames output from MIDL compile step that isn't
quietable via command line flags.
"""
env = self._GetEnv(arch)
args = ['midl', '/nologo'] + list(flags) + [
'/out', outdir,
'/h', h,
idl]
popen = subprocess.Popen(args, shell=True, env=env,
stdout=subprocess.PIPE, stderr=subprocess.STDOUT,
universal_newlines=True)
out, _ = popen.communicate()
# Filter junk out of stdout, and write filtered versions. Output we want
# to filter is pairs of lines that look like this:
# Processing C:\Program Files (x86)\Microsoft SDKs\...\include\objidl.idl
# objidl.idl
lines = out.splitlines()
prefixes = ['Processing ', '64 bit Processing ', '64 bit MIDLRT Processing ']
processing = set()
for prefix in prefixes:
processing = processing.union(set(os.path.basename(x) for x in lines if x.startswith(prefix)))
for line in lines:
if line in processing:
continue
if reduce(lambda x, y : x or y,
[line.startswith(prefix) for prefix in prefixes]):
continue
print line
return popen.returncode
def ExecAsmWrapper(self, arch, *args):
"""Filter logo banner from invocations of asm.exe."""
env = self._GetEnv(arch)
# MSVS doesn't assemble x64 asm files.
if arch == 'environment.x64':
return 0
popen = subprocess.Popen(args, shell=True, env=env,
stdout=subprocess.PIPE, stderr=subprocess.STDOUT,
universal_newlines=True)
out, _ = popen.communicate()
for line in out.splitlines():
if (not line.startswith('Copyright (C) Microsoft Corporation') and
not line.startswith('Microsoft (R) Macro Assembler') and
not line.startswith(' Assembling: ') and
line):
print line
return popen.returncode
def ExecRcWrapper(self, arch, *args):
"""Filter logo banner from invocations of rc.exe. Older versions of RC
don't support the /nologo flag."""
env = self._GetEnv(arch)
popen = subprocess.Popen(args, shell=True, env=env,
stdout=subprocess.PIPE, stderr=subprocess.STDOUT,
universal_newlines=True)
out, _ = popen.communicate()
for line in out.splitlines():
if (not line.startswith('Microsoft (R) Windows (R) Resource Compiler') and
not line.startswith('Copyright (C) Microsoft Corporation') and
line):
print line
return popen.returncode
def ExecActionWrapper(self, arch, rspfile, *dir):
"""Runs an action command line from a response file using the environment
for |arch|. If |dir| is supplied, use that as the working directory."""
env = self._GetEnv(arch)
args = open(rspfile).read()
dir = dir[0] if dir else None
# Local functions bind to outmost function arguments.
def RunCmd():
popen = subprocess.Popen(args, shell=True, env=env, cwd=dir,
stdout=subprocess.PIPE, stderr=subprocess.PIPE,
universal_newlines=True)
stdout_str, stderr_str = popen.communicate()
return popen.returncode, stdout_str, stderr_str
def BuildErrorMessage(err_code, stdout, stderr):
dir_str = os.path.abspath('.') if dir is None else os.path.abspath(dir)
msg = 'ERROR while executing\n' + str(args) + '\ncwd=' + dir_str \
+ '\n' + 'Error code: ' + str(err_code) + '\n'
if stdout:
msg += 'STDOUT:\n' + str(stdout) + '\n'
if stderr:
msg += 'STDERR:\n' + str(stderr) + '\n'
msg += '\n'
return msg
# Note that some commands on windows appears flaky. Therefore commands
# will retry once to make the builds more reliable, and if the cmd fails
# twice then the error will be written out to the command line.
err_code, stdout, stderr = RunCmd()
if err_code == 0:
sys.stdout.write(stdout)
sys.stderr.write(stderr)
return err_code
sys.stdout.write(
BuildErrorMessage(err_code, stdout, stderr) + ' retrying...\n')
err_code, stdout, stderr = RunCmd()
if err_code != 0:
sys.stdout.write(BuildErrorMessage(err_code, stdout, stderr))
return err_code
if __name__ == '__main__':
sys.exit(main(sys.argv[1:]))