| #!/usr/bin/python3 |
| |
| |
| import argparse |
| import collections |
| import json |
| import os |
| import subprocess |
| import sys |
| import tempfile |
| |
| |
| class OrderedSet(collections.OrderedDict): |
| |
| def add(self, value): |
| self[value] = True |
| |
| |
| def compile_module(module, sources, settings, extras, tmpdir): |
| output_file_map = {} |
| if settings.whole_module_optimization: |
| output_file_map[''] = { |
| 'object': os.path.join(settings.object_dir, module + '.o'), |
| 'dependencies': os.path.join(tmpdir, module + '.d'), |
| } |
| else: |
| for source in sources: |
| name, _ = os.path.splitext(os.path.basename(source)) |
| output_file_map[source] = { |
| 'object': os.path.join(settings.object_dir, name + '.o'), |
| 'dependencies': os.path.join(tmpdir, name + '.d'), |
| } |
| |
| for key in ('module_path', 'header_path', 'depfile'): |
| path = getattr(settings, key) |
| if os.path.exists(path): |
| os.unlink(path) |
| if key == 'module_path': |
| for ext in '.swiftdoc', '.swiftsourceinfo': |
| path = os.path.splitext(getattr(settings, key))[0] + ext |
| if os.path.exists(path): |
| os.unlink(path) |
| directory = os.path.dirname(path) |
| if not os.path.exists(directory): |
| os.makedirs(directory) |
| |
| if not os.path.exists(settings.object_dir): |
| os.makedirs(settings.object_dir) |
| |
| for key in output_file_map: |
| path = output_file_map[key]['object'] |
| if os.path.exists(path): |
| os.unlink(path) |
| |
| output_file_map_path = os.path.join(tmpdir, module + '.json') |
| with open(output_file_map_path, 'w') as output_file_map_file: |
| output_file_map_file.write(json.dumps(output_file_map)) |
| output_file_map_file.flush() |
| |
| extra_args = [] |
| if settings.bridge_header: |
| extra_args.extend([ |
| '-import-objc-header', |
| os.path.abspath(settings.bridge_header), |
| ]) |
| |
| if settings.whole_module_optimization: |
| extra_args.append('-whole-module-optimization') |
| |
| if settings.target: |
| extra_args.extend([ |
| '-target', |
| settings.target, |
| ]) |
| |
| if settings.sdk: |
| extra_args.extend([ |
| '-sdk', |
| os.path.abspath(settings.sdk), |
| ]) |
| |
| if settings.include_dirs: |
| for include_dir in settings.include_dirs: |
| extra_args.append('-I' + include_dir) |
| |
| process = subprocess.Popen( |
| ['swiftc', |
| '-parse-as-library', |
| '-module-name', |
| module, |
| '-emit-object', |
| '-emit-dependencies', |
| '-emit-module', |
| '-emit-module-path', |
| settings.module_path, |
| '-emit-objc-header', |
| '-emit-objc-header-path', |
| settings.header_path, |
| '-output-file-map', |
| output_file_map_path, |
| ] + extra_args + extras + sources, |
| stdout=subprocess.PIPE, stderr=subprocess.PIPE, |
| universal_newlines=True) |
| |
| stdout, stderr = process.communicate() |
| if process.returncode: |
| sys.stdout.write(stdout) |
| sys.stderr.write(stderr) |
| sys.exit(process.returncode) |
| |
| |
| depfile_content = collections.OrderedDict() |
| for key in output_file_map: |
| for line in open(output_file_map[key]['dependencies']): |
| output, inputs = line.split(' : ', 2) |
| _, ext = os.path.splitext(output) |
| if ext == '.o': |
| key = output |
| else: |
| key = os.path.splitext(settings.module_path)[0] + ext |
| if key not in depfile_content: |
| depfile_content[key] = OrderedSet() |
| for path in inputs.split(): |
| depfile_content[key].add(path) |
| |
| with open(settings.depfile, 'w') as depfile: |
| for key in depfile_content: |
| if not settings.depfile_filter or key in settings.depfile_filter: |
| inputs = depfile_content[key] |
| depfile.write('%s : %s\n' % (key, ' '.join(inputs))) |
| |
| |
| def main(args): |
| parser = argparse.ArgumentParser(add_help=False) |
| parser.add_argument( |
| '--module-name', |
| help='name of the Swift module') |
| parser.add_argument( |
| '--include', '-I', action='append', dest='include_dirs', |
| help='add directory to header search path') |
| parser.add_argument( |
| 'sources', nargs='+', |
| help='Swift source file to compile') |
| parser.add_argument( |
| '--whole-module-optimization', action='store_true', |
| help='enable whole module optimization') |
| parser.add_argument( |
| '--object-dir', '-o', |
| help='path to the generated object files directory') |
| parser.add_argument( |
| '--module-path', '-m', |
| help='path to the generated module file') |
| parser.add_argument( |
| '--header-path', '-h', |
| help='path to the generated header file') |
| parser.add_argument( |
| '--bridge-header', '-b', |
| help='path to the Objective-C bridge header') |
| parser.add_argument( |
| '--depfile', '-d', |
| help='path to the generated depfile') |
| parser.add_argument( |
| '--depfile-filter', action='append', |
| help='limit depfile to those files') |
| parser.add_argument( |
| '--target', action='store', |
| help='generate code for the given target <triple>') |
| parser.add_argument( |
| '--sdk', action='store', |
| help='compile against sdk') |
| |
| parsed, extras = parser.parse_known_args(args) |
| with tempfile.TemporaryDirectory() as tmpdir: |
| compile_module( |
| parsed.module_name, |
| parsed.sources, |
| parsed, |
| extras, |
| tmpdir) |
| |
| |
| if __name__ == '__main__': |
| sys.exit(main(sys.argv[1:])) |