"""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 subprocess
import sys
import shutil
import json
import wrapper_utils
def collect_soname(args):
"""Replaces: readelf -d $sofile | grep SONAME"""
toc = ''
readelf = subprocess.Popen(wrapper_utils.command_to_run(
[args.readelf, '-d', args.sofile]),
stdout=subprocess.PIPE,
bufsize=-1)
for line in readelf.stdout:
if b'SONAME' in line:
toc += line.decode()
return readelf.wait(), toc
def collect_dyn_sym(args):
"""Replaces: nm --format=posix -g -D $sofile | cut -f1-2 -d' '"""
toc = ''
_command = [args.nm]
if args.sofile.endswith('.dll'):
_command.append('--extern-only')
else:
_command.extend(['--format=posix', '-g', '-D'])
_command.append(args.sofile)
nm = subprocess.Popen(wrapper_utils.command_to_run(_command),
stdout=subprocess.PIPE,
bufsize=-1)
for line in nm.stdout:
toc += '{}\n'.format(' '.join(line.decode().split(' ', 2)[:2]))
return nm.wait(), toc
def collect_toc(args):
result, toc = collect_soname(args)
if result == 0:
result, dynsym = collect_dyn_sym(args)
toc += dynsym
return result, toc
def update_toc(tocfile, toc):
if os.path.exists(tocfile):
with open(tocfile, 'r') as f:
old_toc = f.read()
else:
old_toc = None
if toc != old_toc:
with open(tocfile, 'w') as fp:
fp.write(toc)
def reformat_rsp_file(rspfile):
""" Move all implibs from --whole-archive section"""
with open(rspfile, "r") as fi:
rspcontent = fi.read()
result = []
implibs = []
naflag = False
for arg in rspcontent.split(" "):
if naflag and arg.endswith(".lib"):
implibs.append(arg)
continue
result.append(arg)
if arg == "-Wl,--whole-archive":
naflag = True
continue
if arg == "-Wl,--no-whole-archive":
naflag = False
result.extend(implibs)
with open(rspfile, "w") as fo:
fo.write(" ".join(result))
def get_install_dest_dir(args):
file_list = os.listdir(args.target_out_dir)
target_name = args.target_name
match_file = f'{target_name}_module_info.json'
if not f'{target_name}_module_info.json' in file_list:
return None
with open(os.path.join(args.target_out_dir, match_file), "r") as f:
module_info = json.load(f)
dest_dirs = module_info.get("dest")
for dest_dir in dest_dirs:
if dest_dir.startswith("system"):
return dest_dir
return None
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('--strip-debug-whitelist',
help='The strip debug whitelist, lines of which are names of shared objects with .symtab kept.',
metavar='PATH')
parser.add_argument('--sofile',
required=True,
help='Shared object file produced by linking command',
metavar='FILE')
parser.add_argument('--tocfile',
required=False,
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('--libfile', required=False, metavar='FILE')
parser.add_argument('command', nargs='+', help='Linking command')
parser.add_argument('--mini-debug',
action='store_true',
default=False,
help='Add .gnu_debugdata section for stripped sofile')
parser.add_argument('--target-name', help='')
parser.add_argument('--target-out-dir', help='')
parser.add_argument('--allowed-lib-list', help='')
parser.add_argument('--clang-base-dir', help='')
args = parser.parse_args()
if args.sofile.endswith(".dll"):
rspfile = None
for a in args.command:
if a[0] == "@":
rspfile = a[1:]
break
if rspfile:
reformat_rsp_file(rspfile)
fast_env = dict(os.environ)
fast_env['LC_ALL'] = 'C'
command = wrapper_utils.command_to_run(args.command)
result = wrapper_utils.run_link_with_optional_map_file(
command, env=fast_env, map_file=args.map_file)
if result != 0:
return result
result, toc = collect_toc(args)
if result != 0:
return result
if args.tocfile:
update_toc(args.tocfile, toc)
if args.strip:
strip_option = []
strip_command = [args.strip, '-o', args.output]
if args.target_out_dir and os.path.exists(args.target_out_dir):
install_dest = get_install_dest_dir(args)
else:
install_dest = None
if install_dest:
with open(args.allowed_lib_list, 'r') as f:
lines = f.readlines()
lines = [line.strip()[1:] for line in lines]
if install_dest in lines:
strip_option.extend(['-S'])
if args.strip_debug_whitelist:
with open(args.strip_debug_whitelist, 'r') as whitelist:
for strip_debug_sofile in whitelist.readlines():
if args.sofile.endswith(strip_debug_sofile.strip()):
strip_option.extend(['--strip-debug', '-R', '.comment'])
break
strip_command.extend(strip_option)
strip_command.append(args.sofile)
result = subprocess.call(
wrapper_utils.command_to_run(strip_command))
if args.libfile:
libfile_name = os.path.basename(args.libfile)
sofile_output_dir = os.path.dirname(args.sofile)
unstripped_libfile = os.path.join(sofile_output_dir, libfile_name)
shutil.copy2(unstripped_libfile, args.libfile)
if args.mini_debug and not args.sofile.endswith(".dll"):
unstripped_libfile = os.path.abspath(args.sofile)
script_path = os.path.join(
os.path.dirname(__file__), 'mini_debug_info.py')
ohos_root_path = os.path.join(os.path.dirname(__file__), '../..')
result = subprocess.call(
wrapper_utils.command_to_run(
['python3', script_path, '--unstripped-path', unstripped_libfile, '--stripped-path', args.output,
'--root-path', ohos_root_path, '--clang-base-dir', args.clang_base_dir]))
return result
if __name__ == "__main__":
sys.exit(main())