2015-02-03 09:50:39 +08:00
|
|
|
#!/usr/bin/python
|
|
|
|
|
|
|
|
from __future__ import print_function
|
|
|
|
|
|
|
|
import argparse
|
|
|
|
import getpass
|
|
|
|
import os
|
|
|
|
import os.path
|
|
|
|
import re
|
|
|
|
import select
|
|
|
|
import sys
|
|
|
|
import subprocess
|
|
|
|
|
|
|
|
_COMMON_SYNC_OPTS = "-avzh --delete"
|
|
|
|
_COMMON_EXCLUDE_OPTS = "--exclude=DerivedData --exclude=.svn --exclude=.git --exclude=llvm-build/Release+Asserts"
|
|
|
|
|
2016-09-07 04:57:50 +08:00
|
|
|
|
2015-02-03 09:50:39 +08:00
|
|
|
def normalize_configuration(config_text):
|
|
|
|
if not config_text:
|
|
|
|
return "debug"
|
|
|
|
|
|
|
|
config_lower = config_text.lower()
|
|
|
|
if config_lower in ["debug", "release"]:
|
|
|
|
return config_lower
|
|
|
|
else:
|
|
|
|
raise Exception("unknown configuration specified: %s" % config_text)
|
|
|
|
|
2016-09-07 04:57:50 +08:00
|
|
|
|
2015-02-03 09:50:39 +08:00
|
|
|
def parse_args():
|
|
|
|
DEFAULT_REMOTE_ROOT_DIR = "/mnt/ssd/work/macosx.sync"
|
|
|
|
DEFAULT_REMOTE_HOSTNAME = "tfiala2.mtv.corp.google.com"
|
|
|
|
OPTIONS_FILENAME = ".remote-build.conf"
|
|
|
|
DEFAULT_SSH_PORT = "22"
|
|
|
|
|
|
|
|
parser = argparse.ArgumentParser(fromfile_prefix_chars='@')
|
|
|
|
|
|
|
|
parser.add_argument(
|
2016-09-07 04:57:50 +08:00
|
|
|
"--configuration",
|
|
|
|
"-c",
|
2015-02-03 09:50:39 +08:00
|
|
|
help="specify configuration (Debug, Release)",
|
2016-09-07 04:57:50 +08:00
|
|
|
default=normalize_configuration(
|
|
|
|
os.environ.get(
|
|
|
|
'CONFIGURATION',
|
|
|
|
'Debug')))
|
2015-02-03 09:50:39 +08:00
|
|
|
parser.add_argument(
|
|
|
|
"--debug", "-d",
|
|
|
|
action="store_true",
|
|
|
|
help="help debug the remote-build script by adding extra logging")
|
|
|
|
parser.add_argument(
|
|
|
|
"--local-lldb-dir", "-l", metavar="DIR",
|
|
|
|
help="specify local lldb directory (Xcode layout assumed for llvm/clang)",
|
|
|
|
default=os.getcwd())
|
|
|
|
parser.add_argument(
|
|
|
|
"--port", "-p",
|
|
|
|
help="specify the port ssh should use to connect to the remote side",
|
|
|
|
default=DEFAULT_SSH_PORT)
|
|
|
|
parser.add_argument(
|
|
|
|
"--remote-address", "-r", metavar="REMOTE-ADDR",
|
|
|
|
help="specify the dns name or ip address of the remote linux system",
|
|
|
|
default=DEFAULT_REMOTE_HOSTNAME)
|
|
|
|
parser.add_argument(
|
|
|
|
"--remote-dir", metavar="DIR",
|
|
|
|
help="specify the root of the linux source/build dir",
|
|
|
|
default=DEFAULT_REMOTE_ROOT_DIR)
|
|
|
|
parser.add_argument(
|
|
|
|
"--user", "-u", help="specify the user name for the remote system",
|
|
|
|
default=getpass.getuser())
|
|
|
|
parser.add_argument(
|
2016-09-07 04:57:50 +08:00
|
|
|
"--xcode-action",
|
|
|
|
"-x",
|
|
|
|
help="$(ACTION) from Xcode",
|
|
|
|
nargs='?',
|
|
|
|
default=None)
|
2015-02-03 09:50:39 +08:00
|
|
|
|
|
|
|
command_line_args = sys.argv[1:]
|
|
|
|
if os.path.exists(OPTIONS_FILENAME):
|
2016-09-07 04:57:50 +08:00
|
|
|
# Prepend the file so that command line args override the file
|
|
|
|
# contents.
|
2015-02-03 09:50:39 +08:00
|
|
|
command_line_args.insert(0, "@%s" % OPTIONS_FILENAME)
|
|
|
|
|
|
|
|
return parser.parse_args(command_line_args)
|
|
|
|
|
|
|
|
|
|
|
|
def maybe_create_remote_root_dir(args):
|
|
|
|
commandline = [
|
|
|
|
"ssh",
|
|
|
|
"-p", args.port,
|
|
|
|
"%s@%s" % (args.user, args.remote_address),
|
|
|
|
"mkdir",
|
|
|
|
"-p",
|
|
|
|
args.remote_dir]
|
|
|
|
print("create remote root dir command:\n{}".format(commandline))
|
|
|
|
return subprocess.call(commandline)
|
|
|
|
|
|
|
|
|
|
|
|
def init_with_args(args):
|
|
|
|
# Expand any user directory specs in local-side source dir (on MacOSX).
|
|
|
|
args.local_lldb_dir = os.path.expanduser(args.local_lldb_dir)
|
|
|
|
|
|
|
|
# Append the configuration type to the remote build dir.
|
|
|
|
args.configuration = normalize_configuration(args.configuration)
|
|
|
|
args.remote_build_dir = os.path.join(
|
|
|
|
args.remote_dir,
|
|
|
|
"build-%s" % args.configuration)
|
|
|
|
|
|
|
|
# We assume the local lldb directory is really named 'lldb'.
|
|
|
|
# This is because on the remote end, the local lldb root dir
|
|
|
|
# is copied over underneath llvm/tools and will be named there
|
|
|
|
# whatever it is named locally. The remote build will assume
|
|
|
|
# is is called lldb.
|
|
|
|
if os.path.basename(args.local_lldb_dir) != 'lldb':
|
|
|
|
raise Exception(
|
|
|
|
"local lldb root needs to be called 'lldb' but was {} instead"
|
|
|
|
.format(os.path.basename(args.local_lldb_dir)))
|
|
|
|
|
2016-09-07 04:57:50 +08:00
|
|
|
args.lldb_dir_relative_regex = re.compile(
|
|
|
|
"%s/llvm/tools/lldb/" % args.remote_dir)
|
2015-02-03 09:50:39 +08:00
|
|
|
args.llvm_dir_relative_regex = re.compile("%s/" % args.remote_dir)
|
|
|
|
|
|
|
|
print("Xcode action:", args.xcode_action)
|
|
|
|
|
|
|
|
# Ensure the remote directory exists.
|
|
|
|
result = maybe_create_remote_root_dir(args)
|
|
|
|
if result == 0:
|
|
|
|
print("using remote root dir: %s" % args.remote_dir)
|
|
|
|
else:
|
|
|
|
print("remote root dir doesn't exist and could not be created, "
|
|
|
|
+ "error code:", result)
|
|
|
|
return False
|
|
|
|
|
|
|
|
return True
|
|
|
|
|
2016-09-07 04:57:50 +08:00
|
|
|
|
2015-02-03 09:50:39 +08:00
|
|
|
def sync_llvm(args):
|
|
|
|
commandline = ["rsync"]
|
|
|
|
commandline.extend(_COMMON_SYNC_OPTS.split())
|
|
|
|
commandline.extend(_COMMON_EXCLUDE_OPTS.split())
|
|
|
|
commandline.append("--exclude=/llvm/tools/lldb")
|
|
|
|
commandline.extend(["-e", "ssh -p {}".format(args.port)])
|
|
|
|
commandline.extend([
|
|
|
|
"%s/llvm" % args.local_lldb_dir,
|
|
|
|
"%s@%s:%s" % (args.user, args.remote_address, args.remote_dir)])
|
|
|
|
if args.debug:
|
|
|
|
print("going to execute llvm sync: {}".format(commandline))
|
|
|
|
return subprocess.call(commandline)
|
|
|
|
|
|
|
|
|
|
|
|
def sync_lldb(args):
|
|
|
|
commandline = ["rsync"]
|
|
|
|
commandline.extend(_COMMON_SYNC_OPTS.split())
|
|
|
|
commandline.extend(_COMMON_EXCLUDE_OPTS.split())
|
|
|
|
commandline.append("--exclude=/lldb/llvm")
|
|
|
|
commandline.extend(["-e", "ssh -p {}".format(args.port)])
|
2016-09-07 04:57:50 +08:00
|
|
|
commandline.extend([args.local_lldb_dir, "%s@%s:%s/llvm/tools" %
|
|
|
|
(args.user, args.remote_address, args.remote_dir)])
|
2015-02-03 09:50:39 +08:00
|
|
|
if args.debug:
|
|
|
|
print("going to execute lldb sync: {}".format(commandline))
|
|
|
|
return subprocess.call(commandline)
|
|
|
|
|
|
|
|
|
|
|
|
def build_cmake_command(args):
|
|
|
|
# args.remote_build_dir
|
|
|
|
# args.configuration in ('release', 'debug')
|
|
|
|
|
|
|
|
if args.configuration == 'debug-optimized':
|
|
|
|
build_type_name = "RelWithDebInfo"
|
|
|
|
elif args.configuration == 'release':
|
|
|
|
build_type_name = "Release"
|
|
|
|
else:
|
|
|
|
build_type_name = "Debug"
|
|
|
|
|
|
|
|
ld_flags = "\"-lstdc++ -lm\""
|
|
|
|
|
|
|
|
install_dir = os.path.join(
|
|
|
|
args.remote_build_dir, "..", "install-{}".format(args.configuration))
|
|
|
|
|
|
|
|
command_line = [
|
|
|
|
"cmake",
|
|
|
|
"-GNinja",
|
|
|
|
"-DCMAKE_CXX_COMPILER=clang",
|
|
|
|
"-DCMAKE_C_COMPILER=clang",
|
|
|
|
# "-DCMAKE_CXX_FLAGS=%s" % cxx_flags,
|
|
|
|
"-DCMAKE_SHARED_LINKER_FLAGS=%s" % ld_flags,
|
|
|
|
"-DCMAKE_EXE_LINKER_FLAGS=%s" % ld_flags,
|
|
|
|
"-DCMAKE_INSTALL_PREFIX:PATH=%s" % install_dir,
|
|
|
|
"-DCMAKE_BUILD_TYPE=%s" % build_type_name,
|
|
|
|
"-Wno-dev",
|
|
|
|
os.path.join("..", "llvm")
|
2016-09-07 04:57:50 +08:00
|
|
|
]
|
2015-02-03 09:50:39 +08:00
|
|
|
|
|
|
|
return command_line
|
|
|
|
|
|
|
|
|
|
|
|
def maybe_configure(args):
|
|
|
|
commandline = [
|
|
|
|
"ssh",
|
|
|
|
"-p", args.port,
|
|
|
|
"%s@%s" % (args.user, args.remote_address),
|
|
|
|
"cd", args.remote_dir, "&&",
|
|
|
|
"mkdir", "-p", args.remote_build_dir, "&&",
|
|
|
|
"cd", args.remote_build_dir, "&&"
|
2016-09-07 04:57:50 +08:00
|
|
|
]
|
2015-02-03 09:50:39 +08:00
|
|
|
commandline.extend(build_cmake_command(args))
|
|
|
|
|
|
|
|
if args.debug:
|
|
|
|
print("configure command: {}".format(commandline))
|
|
|
|
|
|
|
|
return subprocess.call(commandline)
|
|
|
|
|
|
|
|
|
|
|
|
def filter_build_line(args, line):
|
|
|
|
lldb_relative_line = args.lldb_dir_relative_regex.sub('', line)
|
|
|
|
if len(lldb_relative_line) != len(line):
|
|
|
|
# We substituted - return the modified line
|
|
|
|
return lldb_relative_line
|
|
|
|
|
|
|
|
# No match on lldb path (longer on linux than llvm path). Try
|
|
|
|
# the llvm path match.
|
|
|
|
return args.llvm_dir_relative_regex.sub('', line)
|
|
|
|
|
|
|
|
|
|
|
|
def run_remote_build_command(args, build_command_list):
|
|
|
|
commandline = [
|
|
|
|
"ssh",
|
|
|
|
"-p", args.port,
|
|
|
|
"%s@%s" % (args.user, args.remote_address),
|
|
|
|
"cd", args.remote_build_dir, "&&"]
|
|
|
|
commandline.extend(build_command_list)
|
|
|
|
|
|
|
|
if args.debug:
|
|
|
|
print("running remote build command: {}".format(commandline))
|
|
|
|
|
|
|
|
proc = subprocess.Popen(
|
|
|
|
commandline,
|
|
|
|
stdout=subprocess.PIPE,
|
|
|
|
stderr=subprocess.PIPE)
|
|
|
|
|
|
|
|
# Filter stdout/stderr output for file path mapping.
|
|
|
|
# We do this to enable Xcode to see filenames relative to the
|
|
|
|
# MacOSX-side directory structure.
|
|
|
|
while True:
|
|
|
|
reads = [proc.stdout.fileno(), proc.stderr.fileno()]
|
|
|
|
select_result = select.select(reads, [], [])
|
|
|
|
|
|
|
|
for fd in select_result[0]:
|
|
|
|
if fd == proc.stdout.fileno():
|
|
|
|
line = proc.stdout.readline()
|
|
|
|
display_line = filter_build_line(args, line.rstrip())
|
|
|
|
if display_line and len(display_line) > 0:
|
|
|
|
print(display_line)
|
|
|
|
elif fd == proc.stderr.fileno():
|
|
|
|
line = proc.stderr.readline()
|
|
|
|
display_line = filter_build_line(args, line.rstrip())
|
|
|
|
if display_line and len(display_line) > 0:
|
|
|
|
print(display_line, file=sys.stderr)
|
|
|
|
|
|
|
|
proc_retval = proc.poll()
|
2016-09-07 04:57:50 +08:00
|
|
|
if proc_retval is not None:
|
2015-02-03 09:50:39 +08:00
|
|
|
# Process stopped. Drain output before finishing up.
|
|
|
|
|
|
|
|
# Drain stdout.
|
|
|
|
while True:
|
|
|
|
line = proc.stdout.readline()
|
|
|
|
if line:
|
|
|
|
display_line = filter_build_line(args, line.rstrip())
|
|
|
|
if display_line and len(display_line) > 0:
|
|
|
|
print(display_line)
|
|
|
|
else:
|
|
|
|
break
|
|
|
|
|
|
|
|
# Drain stderr.
|
|
|
|
while True:
|
|
|
|
line = proc.stderr.readline()
|
|
|
|
if line:
|
|
|
|
display_line = filter_build_line(args, line.rstrip())
|
|
|
|
if display_line and len(display_line) > 0:
|
|
|
|
print(display_line, file=sys.stderr)
|
|
|
|
else:
|
|
|
|
break
|
|
|
|
|
|
|
|
return proc_retval
|
|
|
|
|
|
|
|
|
|
|
|
def build(args):
|
|
|
|
return run_remote_build_command(args, ["time", "ninja"])
|
|
|
|
|
|
|
|
|
|
|
|
def clean(args):
|
|
|
|
return run_remote_build_command(args, ["ninja", "clean"])
|
|
|
|
|
|
|
|
|
|
|
|
if __name__ == "__main__":
|
|
|
|
# Handle arg parsing.
|
|
|
|
args = parse_args()
|
|
|
|
|
|
|
|
# Initialize the system.
|
|
|
|
if not init_with_args(args):
|
|
|
|
exit(1)
|
|
|
|
|
|
|
|
# Sync over llvm and clang source.
|
|
|
|
sync_llvm(args)
|
|
|
|
|
|
|
|
# Sync over lldb source.
|
|
|
|
sync_lldb(args)
|
|
|
|
|
|
|
|
# Configure the remote build if it's not already.
|
|
|
|
maybe_configure(args)
|
|
|
|
|
|
|
|
if args.xcode_action == 'clean':
|
|
|
|
exit(clean(args))
|
|
|
|
else:
|
|
|
|
exit(build(args))
|