1#!/usr/bin/env python
2# Copyright 2016 the V8 project authors. All rights reserved.
3# Use of this source code is governed by a BSD-style license that can be
4# found in the LICENSE file.
5
6"""Script to generate V8's gn arguments based on common developer defaults
7or builder configurations.
8
9Goma is used by default if detected. The compiler proxy is assumed to run.
10
11This script can be added to the PATH and be used on other checkouts. It always
12runs for the checkout nesting the CWD.
13
14Configurations of this script live in infra/mb/mb_config.pyl.
15
16Available actions are: {gen,list}. Omitting the action defaults to "gen".
17
18-------------------------------------------------------------------------------
19
20Examples:
21
22# Generate the ia32.release config in out.gn/ia32.release.
23v8gen.py ia32.release
24
25# Generate into out.gn/foo without goma auto-detect.
26v8gen.py gen -b ia32.release foo --no-goma
27
28# Pass additional gn arguments after -- (don't use spaces within gn args).
29v8gen.py ia32.optdebug -- v8_enable_slow_dchecks=true
30
31# Generate gn arguments of 'V8 Linux64 - builder' from 'client.v8'. To switch
32# off goma usage here, the args.gn file must be edited manually.
33v8gen.py -m client.v8 -b 'V8 Linux64 - builder'
34
35# Show available configurations.
36v8gen.py list
37
38-------------------------------------------------------------------------------
39"""
40
41import argparse
42import os
43import re
44import subprocess
45import sys
46
47CONFIG = os.path.join('infra', 'mb', 'mb_config.pyl')
48GOMA_DEFAULT = os.path.join(os.path.expanduser("~"), 'goma')
49OUT_DIR = 'out.gn'
50
51TOOLS_PATH = os.path.dirname(os.path.dirname(os.path.abspath(__file__)))
52sys.path.append(os.path.join(TOOLS_PATH, 'mb'))
53
54import mb
55
56
57def _sanitize_nonalpha(text):
58  return re.sub(r'[^a-zA-Z0-9.]', '_', text)
59
60
61class GenerateGnArgs(object):
62  def __init__(self, args):
63    # Split args into this script's arguments and gn args passed to the
64    # wrapped gn.
65    index = args.index('--') if '--' in args else len(args)
66    self._options = self._parse_arguments(args[:index])
67    self._gn_args = args[index + 1:]
68
69  def _parse_arguments(self, args):
70    self.parser = argparse.ArgumentParser(
71      description=__doc__,
72      formatter_class=argparse.RawTextHelpFormatter,
73    )
74
75    def add_common_options(p):
76      p.add_argument(
77          '-m', '--master', default='developer_default',
78          help='config group or master from mb_config.pyl - default: '
79               'developer_default')
80      p.add_argument(
81          '-v', '--verbosity', action='count',
82          help='print wrapped commands (use -vv to print output of wrapped '
83               'commands)')
84
85    subps = self.parser.add_subparsers()
86
87    # Command: gen.
88    gen_cmd = subps.add_parser(
89        'gen', help='generate a new set of build files (default)')
90    gen_cmd.set_defaults(func=self.cmd_gen)
91    add_common_options(gen_cmd)
92    gen_cmd.add_argument(
93        'outdir', nargs='?',
94        help='optional gn output directory')
95    gen_cmd.add_argument(
96        '-b', '--builder',
97        help='build configuration or builder name from mb_config.pyl, e.g. '
98             'x64.release')
99    gen_cmd.add_argument(
100        '-p', '--pedantic', action='store_true',
101        help='run gn over command-line gn args to catch errors early')
102
103    goma = gen_cmd.add_mutually_exclusive_group()
104    goma.add_argument(
105        '-g' , '--goma',
106        action='store_true', default=None, dest='goma',
107        help='force using goma')
108    goma.add_argument(
109        '--nogoma', '--no-goma',
110        action='store_false', default=None, dest='goma',
111        help='don\'t use goma auto detection - goma might still be used if '
112             'specified as a gn arg')
113
114    # Command: list.
115    list_cmd = subps.add_parser(
116        'list', help='list available configurations')
117    list_cmd.set_defaults(func=self.cmd_list)
118    add_common_options(list_cmd)
119
120    # Default to "gen" unless global help is requested.
121    if not args or args[0] not in subps.choices.keys() + ['-h', '--help']:
122      args = ['gen'] + args
123
124    return self.parser.parse_args(args)
125
126  def cmd_gen(self):
127    if not self._options.outdir and not self._options.builder:
128      self.parser.error('please specify either an output directory or '
129                        'a builder/config name (-b), e.g. x64.release')
130
131    if not self._options.outdir:
132      # Derive output directory from builder name.
133      self._options.outdir = _sanitize_nonalpha(self._options.builder)
134    else:
135      # Also, if this should work on windows, we might need to use \ where
136      # outdir is used as path, while using / if it's used in a gn context.
137      if self._options.outdir.startswith('/'):
138        self.parser.error(
139            'only output directories relative to %s are supported' % OUT_DIR)
140
141    if not self._options.builder:
142      # Derive builder from output directory.
143      self._options.builder = self._options.outdir
144
145    # Check for builder/config in mb config.
146    if self._options.builder not in self._mbw.masters[self._options.master]:
147      print '%s does not exist in %s for %s' % (
148          self._options.builder, CONFIG, self._options.master)
149      return 1
150
151    # TODO(machenbach): Check if the requested configurations has switched to
152    # gn at all.
153
154    # The directories are separated with slashes in a gn context (platform
155    # independent).
156    gn_outdir = '/'.join([OUT_DIR, self._options.outdir])
157
158    # Call MB to generate the basic configuration.
159    self._call_cmd([
160      sys.executable,
161      '-u', os.path.join('tools', 'mb', 'mb.py'),
162      'gen',
163      '-f', CONFIG,
164      '-m', self._options.master,
165      '-b', self._options.builder,
166      gn_outdir,
167    ])
168
169    # Handle extra gn arguments.
170    gn_args_path = os.path.join(OUT_DIR, self._options.outdir, 'args.gn')
171
172    # Append command-line args.
173    modified = self._append_gn_args(
174        'command-line', gn_args_path, '\n'.join(self._gn_args))
175
176    # Append goma args.
177    # TODO(machenbach): We currently can't remove existing goma args from the
178    # original config. E.g. to build like a bot that uses goma, but switch
179    # goma off.
180    modified |= self._append_gn_args(
181        'goma', gn_args_path, self._goma_args)
182
183    # Regenerate ninja files to check for errors in the additional gn args.
184    if modified and self._options.pedantic:
185      self._call_cmd(['gn', 'gen', gn_outdir])
186    return 0
187
188  def cmd_list(self):
189    print '\n'.join(sorted(self._mbw.masters[self._options.master]))
190    return 0
191
192  def verbose_print_1(self, text):
193    if self._options.verbosity >= 1:
194      print '#' * 80
195      print text
196
197  def verbose_print_2(self, text):
198    if self._options.verbosity >= 2:
199      indent = ' ' * 2
200      for l in text.splitlines():
201        print indent + l
202
203  def _call_cmd(self, args):
204    self.verbose_print_1(' '.join(args))
205    try:
206      output = subprocess.check_output(
207        args=args,
208        stderr=subprocess.STDOUT,
209      )
210      self.verbose_print_2(output)
211    except subprocess.CalledProcessError as e:
212      self.verbose_print_2(e.output)
213      raise
214
215  def _find_work_dir(self, path):
216    """Find the closest v8 root to `path`."""
217    if os.path.exists(os.path.join(path, 'tools', 'dev', 'v8gen.py')):
218      # Approximate the v8 root dir by a folder where this script exists
219      # in the expected place.
220      return path
221    elif os.path.dirname(path) == path:
222      raise Exception(
223          'This appears to not be called from a recent v8 checkout')
224    else:
225      return self._find_work_dir(os.path.dirname(path))
226
227  @property
228  def _goma_dir(self):
229    return os.path.normpath(os.environ.get('GOMA_DIR') or GOMA_DEFAULT)
230
231  @property
232  def _need_goma_dir(self):
233    return self._goma_dir != GOMA_DEFAULT
234
235  @property
236  def _use_goma(self):
237    if self._options.goma is None:
238      # Auto-detect.
239      return os.path.exists(self._goma_dir) and os.path.isdir(self._goma_dir)
240    else:
241      return self._options.goma
242
243  @property
244  def _goma_args(self):
245    """Gn args for using goma."""
246    # Specify goma args if we want to use goma and if goma isn't specified
247    # via command line already. The command-line always has precedence over
248    # any other specification.
249    if (self._use_goma and
250        not any(re.match(r'use_goma\s*=.*', x) for x in self._gn_args)):
251      if self._need_goma_dir:
252        return 'use_goma=true\ngoma_dir="%s"' % self._goma_dir
253      else:
254        return 'use_goma=true'
255    else:
256      return ''
257
258  def _append_gn_args(self, type, gn_args_path, more_gn_args):
259    """Append extra gn arguments to the generated args.gn file."""
260    if not more_gn_args:
261      return False
262    self.verbose_print_1('Appending """\n%s\n""" to %s.' % (
263        more_gn_args, os.path.abspath(gn_args_path)))
264    with open(gn_args_path, 'a') as f:
265      f.write('\n# Additional %s args:\n' % type)
266      f.write(more_gn_args)
267      f.write('\n')
268
269    # Artificially increment modification time as our modifications happen too
270    # fast. This makes sure that gn is properly rebuilding the ninja files.
271    mtime = os.path.getmtime(gn_args_path) + 1
272    with open(gn_args_path, 'a'):
273      os.utime(gn_args_path, (mtime, mtime))
274
275    return True
276
277  def main(self):
278    # Always operate relative to the base directory for better relative-path
279    # handling. This script can be used in any v8 checkout.
280    workdir = self._find_work_dir(os.getcwd())
281    if workdir != os.getcwd():
282      self.verbose_print_1('cd ' + workdir)
283      os.chdir(workdir)
284
285    # Initialize MB as a library.
286    self._mbw = mb.MetaBuildWrapper()
287
288    # TODO(machenbach): Factor out common methods independent of mb arguments.
289    self._mbw.ParseArgs(['lookup', '-f', CONFIG])
290    self._mbw.ReadConfigFile()
291
292    if not self._options.master in self._mbw.masters:
293      print '%s not found in %s\n' % (self._options.master, CONFIG)
294      print 'Choose one of:\n%s\n' % (
295          '\n'.join(sorted(self._mbw.masters.keys())))
296      return 1
297
298    return self._options.func()
299
300
301if __name__ == "__main__":
302  gen = GenerateGnArgs(sys.argv[1:])
303  try:
304    sys.exit(gen.main())
305  except Exception:
306    if gen._options.verbosity < 2:
307      print ('\nHint: You can raise verbosity (-vv) to see the output of '
308             'failed commands.\n')
309    raise
310