| #!/usr/bin/env python |
| # Copyright 2015 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. |
| |
| """Runs 'ld -shared' and generates a .TOC file that's untouched when unchanged. |
| |
| This script exists to avoid using complex shell commands in |
| gcc_toolchain.gni's tool("solink"), in case the host running the compiler |
| does not have a POSIX-like shell (e.g. Windows). |
| """ |
| |
| import argparse |
| import os |
| import shlex |
| import subprocess |
| import sys |
| |
| import wrapper_utils |
| |
| |
| def CollectSONAME(args): |
| """Replaces: readelf -d $sofile | grep SONAME""" |
| toc = '' |
| readelf = subprocess.Popen(wrapper_utils.CommandToRun( |
| [args.readelf, '-d', args.sofile]), |
| stdout=subprocess.PIPE, |
| bufsize=-1, |
| universal_newlines=True) |
| for line in readelf.stdout: |
| if 'SONAME' in line: |
| toc += line |
| return readelf.wait(), toc |
| |
| |
| def CollectDynSym(args): |
| """Replaces: nm --format=posix -g -D -p $sofile | cut -f1-2 -d' '""" |
| toc = '' |
| nm = subprocess.Popen(wrapper_utils.CommandToRun( |
| [args.nm, '--format=posix', '-g', '-D', '-p', args.sofile]), |
| stdout=subprocess.PIPE, |
| bufsize=-1, |
| universal_newlines=True) |
| for line in nm.stdout: |
| toc += ' '.join(line.split(' ', 2)[:2]) + '\n' |
| return nm.wait(), toc |
| |
| |
| def CollectTOC(args): |
| result, toc = CollectSONAME(args) |
| if result == 0: |
| result, dynsym = CollectDynSym(args) |
| toc += dynsym |
| return result, toc |
| |
| |
| def UpdateTOC(tocfile, toc): |
| if os.path.exists(tocfile): |
| old_toc = open(tocfile, 'r').read() |
| else: |
| old_toc = None |
| if toc != old_toc: |
| open(tocfile, 'w').write(toc) |
| |
| |
| def CollectInputs(out, args): |
| for x in args: |
| if x.startswith('@'): |
| with open(x[1:]) as rsp: |
| CollectInputs(out, shlex.split(rsp.read())) |
| elif not x.startswith('-') and (x.endswith('.o') or x.endswith('.a')): |
| out.write(x) |
| out.write('\n') |
| |
| |
| def InterceptFlag(flag, command): |
| ret = flag in command |
| if ret: |
| command.remove(flag) |
| return ret |
| |
| |
| def main(): |
| parser = argparse.ArgumentParser(description=__doc__) |
| parser.add_argument('--readelf', |
| required=True, |
| help='The readelf binary to run', |
| metavar='PATH') |
| parser.add_argument('--nm', |
| required=True, |
| help='The nm binary to run', |
| metavar='PATH') |
| parser.add_argument('--strip', |
| help='The strip binary to run', |
| metavar='PATH') |
| parser.add_argument('--dwp', help='The dwp binary to run', metavar='PATH') |
| parser.add_argument('--sofile', |
| required=True, |
| help='Shared object file produced by linking command', |
| metavar='FILE') |
| parser.add_argument('--tocfile', |
| required=True, |
| help='Output table-of-contents file', |
| metavar='FILE') |
| parser.add_argument('--map-file', |
| help=('Use --Wl,-Map to generate a map file. Will be ' |
| 'gzipped if extension ends with .gz'), |
| metavar='FILE') |
| parser.add_argument('--output', |
| required=True, |
| help='Final output shared object file', |
| metavar='FILE') |
| parser.add_argument('command', nargs='+', |
| help='Linking command') |
| args = parser.parse_args() |
| |
| # Work-around for gold being slow-by-default. http://crbug.com/632230 |
| fast_env = dict(os.environ) |
| fast_env['LC_ALL'] = 'C' |
| |
| # Extract flags passed through ldflags but meant for this script. |
| # https://crbug.com/954311 tracks finding a better way to plumb these. |
| link_only = InterceptFlag('--link-only', args.command) |
| collect_inputs_only = InterceptFlag('--collect-inputs-only', args.command) |
| |
| # If only linking, we are likely generating a partitioned .so that will be |
| # split apart later. In that case: |
| # |
| # - The TOC file optimization isn't useful, because the partition libraries |
| # must always be re-extracted if the combined library changes (and nothing |
| # should be depending on the combined library's dynamic symbol table). |
| # - Stripping isn't necessary, because the combined library is not used in |
| # production or published. |
| # |
| # Both of these operations could still be done, they're needless work, and |
| # tools would need to be updated to handle and/or not complain about |
| # partitioned libraries. Instead, to keep Ninja happy, simply create dummy |
| # files for the TOC and stripped lib. |
| if link_only or collect_inputs_only: |
| open(args.output, 'w').close() |
| open(args.tocfile, 'w').close() |
| if args.dwp: |
| open(args.sofile + '.dwp', 'w').close() |
| |
| # Instead of linking, records all inputs to a file. This is used by |
| # enable_resource_allowlist_generation in order to avoid needing to |
| # link (which is slow) to build the resources allowlist. |
| if collect_inputs_only: |
| with open(args.sofile, 'w') as f: |
| CollectInputs(f, args.command) |
| if args.map_file: |
| open(args.map_file, 'w').close() |
| return 0 |
| |
| # First, run the actual link. |
| command = wrapper_utils.CommandToRun(args.command) |
| result = wrapper_utils.RunLinkWithOptionalMapFile(command, |
| env=fast_env, |
| map_file=args.map_file) |
| |
| if result != 0 or link_only: |
| return result |
| |
| # If dwp is set, then package debug info for this SO. |
| dwp_proc = None |
| if args.dwp: |
| # Suppress output here because it doesn't seem to be useful. The most |
| # common error is a segfault, which will happen if files are missing. |
| with open(os.devnull, "w") as devnull: |
| dwp_proc = subprocess.Popen(wrapper_utils.CommandToRun( |
| [args.dwp, '-e', args.sofile, '-o', args.sofile + '.dwp']), |
| stdout=devnull, |
| stderr=subprocess.STDOUT) |
| |
| # Next, generate the contents of the TOC file. |
| result, toc = CollectTOC(args) |
| if result != 0: |
| return result |
| |
| # If there is an existing TOC file with identical contents, leave it alone. |
| # Otherwise, write out the TOC file. |
| UpdateTOC(args.tocfile, toc) |
| |
| # Finally, strip the linked shared object file (if desired). |
| if args.strip: |
| result = subprocess.call(wrapper_utils.CommandToRun( |
| [args.strip, '-o', args.output, args.sofile])) |
| |
| if dwp_proc: |
| dwp_result = dwp_proc.wait() |
| if dwp_result != 0: |
| return dwp_result |
| |
| return result |
| |
| |
| if __name__ == "__main__": |
| sys.exit(main()) |