compile_single_file.py 2.45 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74
#!/usr/bin/env python
# Copyright 2017 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 argparse
import os
import subprocess
import sys


# This function is inspired from the one in src/tools/vim/ninja-build.vim in the
# Chromium repository.
def path_to_source_root(path):
  """Returns the absolute path to the chromium source root."""
  candidate = os.path.dirname(path)
  # This is a list of directories that need to identify the src directory. The
  # shorter it is, the more likely it's wrong (checking for just
  # "build/common.gypi" would find "src/v8" for files below "src/v8", as
  # "src/v8/build/common.gypi" exists). The longer it is, the more likely it is
  # to break when we rename directories.
  fingerprints = ['chrome', 'net', 'v8', 'build', 'skia']
  while candidate and not all(
      [os.path.isdir(os.path.join(candidate, fp)) for fp in fingerprints]):
    new_candidate = os.path.dirname(candidate)
    if new_candidate == candidate:
      raise Exception("Couldn't find source-dir from %s" % path)
    candidate = os.path.dirname(candidate)
  return candidate


def main():
  parser = argparse.ArgumentParser()
  parser.add_argument(
      '--file-path',
      help='The file path, could be absolute or relative to the current '
           'directory.',
      required=True)
  parser.add_argument(
      '--build-dir',
      help='The build directory, relative to the source directory.',
      required=True)

  options = parser.parse_args()

  src_dir = path_to_source_root(os.path.abspath(options.file_path))
  abs_build_dir = os.path.join(src_dir, options.build_dir)
  src_relpath = os.path.relpath(options.file_path, abs_build_dir)

  print 'Building %s' % options.file_path

  ninja_exec = 'ninja'
  carets = '^'
  # We need to make sure that we call the ninja executable, calling |ninja|
  # directly might end up calling a wrapper script that'll remove the caret
  # characters.
  if sys.platform == 'win32':
    ninja_exec = 'ninja.exe'
    # The caret character has to be escaped on Windows as it's an escape
    # character.
    carets = '^^'

  command = [
      ninja_exec,
      '-C', abs_build_dir,
      '%s%s' % (src_relpath, carets)
  ]
  # |shell| should be set to True on Windows otherwise the carets characters
  # get dropped from the command line.
  return subprocess.call(command, shell=sys.platform=='win32')


if __name__ == '__main__':
  sys.exit(main())