|  | #!/usr/bin/env python | 
|  | # | 
|  | # Copyright 2016 Google Inc. | 
|  | # | 
|  | # Use of this source code is governed by a BSD-style license that can be | 
|  | # found in the LICENSE file. | 
|  |  | 
|  | from __future__ import print_function | 
|  |  | 
|  | import collections | 
|  | import json | 
|  | import os | 
|  | import subprocess | 
|  | import sys | 
|  |  | 
|  | # Finds all public sources in include directories then write them to skia.h. | 
|  |  | 
|  | # Also write skia.h.deps, which Ninja uses to track dependencies. It's the | 
|  | # very same mechanism Ninja uses to know which .h files affect which .cpp files. | 
|  |  | 
|  | gn              = sys.argv[1] | 
|  | absolute_source = sys.argv[2] | 
|  | skia_h          = sys.argv[3] | 
|  | include_dirs    = sys.argv[4:] | 
|  |  | 
|  | absolute_source = os.path.normpath(absolute_source) | 
|  |  | 
|  | include_dirs = [os.path.join(os.path.normpath(include_dir), '') | 
|  | for include_dir in include_dirs] | 
|  | include_dirs.sort(key=len, reverse=True) | 
|  |  | 
|  | gn_desc_cmd = [gn, 'desc', '.', '--root=%s' % absolute_source, '--format=json', | 
|  | '*'] | 
|  |  | 
|  | desc_json_txt = '' | 
|  | try: | 
|  | desc_json_txt = subprocess.check_output(gn_desc_cmd).decode('utf-8') | 
|  | except subprocess.CalledProcessError as e: | 
|  | print(e.output.decode('utf-8')) | 
|  | raise | 
|  |  | 
|  | if desc_json_txt.startswith('WARNING'): | 
|  | print('\ngn generated a warning when we asked for JSON output.', | 
|  | 'To see the warning, run this command from the out_dir:', | 
|  | '(you may need to quote the * argument)\n', | 
|  | ' '.join(gn_desc_cmd), | 
|  | '\n', sep='\n') | 
|  | sys.exit(-1) | 
|  |  | 
|  | desc_json = {} | 
|  | try: | 
|  | desc_json = json.loads(desc_json_txt) | 
|  | except ValueError: | 
|  | print(desc_json_txt) | 
|  | raise | 
|  |  | 
|  | sources = set() | 
|  |  | 
|  | for target in desc_json.values(): | 
|  | # We'll use `public` headers if they're listed, or pull them from `sources` | 
|  | # if not.  GN sneaks in a default "public": "*" into the JSON if you don't | 
|  | # set one explicitly. | 
|  | search_list = target.get('public') | 
|  | if search_list == '*': | 
|  | search_list = target.get('sources', []) | 
|  |  | 
|  | for name in search_list: | 
|  | sources.add(os.path.join(absolute_source, os.path.normpath(name[2:]))) | 
|  |  | 
|  | Header = collections.namedtuple('Header', ['absolute', 'include']) | 
|  | headers = {} | 
|  | for source in sources: | 
|  | source_as_include = [os.path.relpath(source, absolute_source) | 
|  | for include_dir in include_dirs | 
|  | if source.startswith(include_dir)] | 
|  | if not source_as_include: | 
|  | continue | 
|  | statinfo = os.stat(source) | 
|  | key = str(statinfo.st_ino) + ':' + str(statinfo.st_dev) | 
|  | # On Windows os.stat st_ino is 0 until 3.3.4 and st_dev is 0 until 3.4.0. | 
|  | if key == '0:0': | 
|  | key = source | 
|  | include_path = source_as_include[0] | 
|  | if key not in headers or len(include_path) < len(headers[key].include): | 
|  | headers[key] = Header(source, include_path) | 
|  |  | 
|  | headers = sorted(headers.values(), key=lambda x: x.include) | 
|  |  | 
|  | with open(skia_h, 'w') as f: | 
|  | f.write('// skia.h generated by GN.\n') | 
|  | f.write('#ifndef skia_h_DEFINED\n') | 
|  | f.write('#define skia_h_DEFINED\n') | 
|  | for header in headers: | 
|  | f.write('#include "' + header.include + '"\n') | 
|  | f.write('#endif//skia_h_DEFINED\n') | 
|  |  | 
|  | with open(skia_h + '.deps', 'w') as f: | 
|  | f.write(skia_h + ':') | 
|  | for header in headers: | 
|  | f.write(' ' + header.absolute) | 
|  | f.write(' build.ninja.d') | 
|  | f.write('\n') | 
|  |  | 
|  | # Temporary: during development this file wrote skia.h.d, not skia.h.deps, | 
|  | # and I think we have some bad versions of those files laying around. | 
|  | if os.path.exists(skia_h + '.d'): | 
|  | os.remove(skia_h + '.d') |