15f9996aaSopenharmony_ci#!/usr/bin/env python
25f9996aaSopenharmony_ci# -*- coding: utf-8 -*-
35f9996aaSopenharmony_ci# Copyright 2015 The Chromium Authors. All rights reserved.
45f9996aaSopenharmony_ci# Use of this source code is governed by a BSD-style license that can be
55f9996aaSopenharmony_ci# found in the LICENSE file.
65f9996aaSopenharmony_ci"""Runs 'ld -shared' and generates a .TOC file that's untouched when unchanged.
75f9996aaSopenharmony_ci
85f9996aaSopenharmony_ciThis script exists to avoid using complex shell commands in
95f9996aaSopenharmony_cigcc_toolchain.gni's tool("solink"), in case the host running the compiler
105f9996aaSopenharmony_cidoes not have a POSIX-like shell (e.g. Windows).
115f9996aaSopenharmony_ci"""
125f9996aaSopenharmony_ci
135f9996aaSopenharmony_ciimport argparse
145f9996aaSopenharmony_ciimport os
155f9996aaSopenharmony_ciimport subprocess
165f9996aaSopenharmony_ciimport sys
175f9996aaSopenharmony_ciimport shutil
185f9996aaSopenharmony_ciimport json
195f9996aaSopenharmony_ci
205f9996aaSopenharmony_ciimport wrapper_utils
215f9996aaSopenharmony_ci
225f9996aaSopenharmony_ci
235f9996aaSopenharmony_cidef collect_soname(args):
245f9996aaSopenharmony_ci    """Replaces: readelf -d $sofile | grep SONAME"""
255f9996aaSopenharmony_ci    toc = ''
265f9996aaSopenharmony_ci    readelf = subprocess.Popen(wrapper_utils.command_to_run(
275f9996aaSopenharmony_ci        [args.readelf, '-d', args.sofile]),
285f9996aaSopenharmony_ci        stdout=subprocess.PIPE,
295f9996aaSopenharmony_ci        bufsize=-1)
305f9996aaSopenharmony_ci    for line in readelf.stdout:
315f9996aaSopenharmony_ci        if b'SONAME' in line:
325f9996aaSopenharmony_ci            toc += line.decode()
335f9996aaSopenharmony_ci    return readelf.wait(), toc
345f9996aaSopenharmony_ci
355f9996aaSopenharmony_ci
365f9996aaSopenharmony_cidef collect_dyn_sym(args):
375f9996aaSopenharmony_ci    """Replaces: nm --format=posix -g -D $sofile | cut -f1-2 -d' '"""
385f9996aaSopenharmony_ci    toc = ''
395f9996aaSopenharmony_ci    _command = [args.nm]
405f9996aaSopenharmony_ci    if args.sofile.endswith('.dll'):
415f9996aaSopenharmony_ci        _command.append('--extern-only')
425f9996aaSopenharmony_ci    else:
435f9996aaSopenharmony_ci        _command.extend(['--format=posix', '-g', '-D'])
445f9996aaSopenharmony_ci    _command.append(args.sofile)
455f9996aaSopenharmony_ci    nm = subprocess.Popen(wrapper_utils.command_to_run(_command),
465f9996aaSopenharmony_ci                          stdout=subprocess.PIPE,
475f9996aaSopenharmony_ci                          bufsize=-1)
485f9996aaSopenharmony_ci    for line in nm.stdout:
495f9996aaSopenharmony_ci        toc += '{}\n'.format(' '.join(line.decode().split(' ', 2)[:2]))
505f9996aaSopenharmony_ci    return nm.wait(), toc
515f9996aaSopenharmony_ci
525f9996aaSopenharmony_ci
535f9996aaSopenharmony_cidef collect_toc(args):
545f9996aaSopenharmony_ci    result, toc = collect_soname(args)
555f9996aaSopenharmony_ci    if result == 0:
565f9996aaSopenharmony_ci        result, dynsym = collect_dyn_sym(args)
575f9996aaSopenharmony_ci        toc += dynsym
585f9996aaSopenharmony_ci    return result, toc
595f9996aaSopenharmony_ci
605f9996aaSopenharmony_ci
615f9996aaSopenharmony_cidef update_toc(tocfile, toc):
625f9996aaSopenharmony_ci    if os.path.exists(tocfile):
635f9996aaSopenharmony_ci        with open(tocfile, 'r') as f:
645f9996aaSopenharmony_ci            old_toc = f.read()
655f9996aaSopenharmony_ci    else:
665f9996aaSopenharmony_ci        old_toc = None
675f9996aaSopenharmony_ci    if toc != old_toc:
685f9996aaSopenharmony_ci        with open(tocfile, 'w') as fp:
695f9996aaSopenharmony_ci            fp.write(toc)
705f9996aaSopenharmony_ci
715f9996aaSopenharmony_ci
725f9996aaSopenharmony_cidef reformat_rsp_file(rspfile):
735f9996aaSopenharmony_ci    """ Move all implibs from --whole-archive section"""
745f9996aaSopenharmony_ci    with open(rspfile, "r") as fi:
755f9996aaSopenharmony_ci        rspcontent = fi.read()
765f9996aaSopenharmony_ci    result = []
775f9996aaSopenharmony_ci    implibs = []
785f9996aaSopenharmony_ci    naflag = False
795f9996aaSopenharmony_ci    for arg in rspcontent.split(" "):
805f9996aaSopenharmony_ci        if naflag and arg.endswith(".lib"):
815f9996aaSopenharmony_ci            implibs.append(arg)
825f9996aaSopenharmony_ci            continue
835f9996aaSopenharmony_ci        result.append(arg)
845f9996aaSopenharmony_ci        if arg == "-Wl,--whole-archive":
855f9996aaSopenharmony_ci            naflag = True
865f9996aaSopenharmony_ci            continue
875f9996aaSopenharmony_ci        if arg == "-Wl,--no-whole-archive":
885f9996aaSopenharmony_ci            naflag = False
895f9996aaSopenharmony_ci            result.extend(implibs)
905f9996aaSopenharmony_ci
915f9996aaSopenharmony_ci    with open(rspfile, "w") as fo:
925f9996aaSopenharmony_ci        fo.write(" ".join(result))
935f9996aaSopenharmony_ci
945f9996aaSopenharmony_ci
955f9996aaSopenharmony_cidef get_install_dest_dir(args):
965f9996aaSopenharmony_ci    file_list = os.listdir(args.target_out_dir)
975f9996aaSopenharmony_ci    target_name = args.target_name
985f9996aaSopenharmony_ci    match_file = f'{target_name}_module_info.json'
995f9996aaSopenharmony_ci    if not f'{target_name}_module_info.json' in file_list:
1005f9996aaSopenharmony_ci        return None
1015f9996aaSopenharmony_ci    with open(os.path.join(args.target_out_dir, match_file), "r") as f:
1025f9996aaSopenharmony_ci        module_info = json.load(f)
1035f9996aaSopenharmony_ci        dest_dirs = module_info.get("dest")
1045f9996aaSopenharmony_ci        for dest_dir in dest_dirs:
1055f9996aaSopenharmony_ci            if dest_dir.startswith("system"):
1065f9996aaSopenharmony_ci                return dest_dir
1075f9996aaSopenharmony_ci        return None
1085f9996aaSopenharmony_ci
1095f9996aaSopenharmony_ci
1105f9996aaSopenharmony_cidef main():
1115f9996aaSopenharmony_ci    parser = argparse.ArgumentParser(description=__doc__)
1125f9996aaSopenharmony_ci    parser.add_argument('--readelf',
1135f9996aaSopenharmony_ci                        required=True,
1145f9996aaSopenharmony_ci                        help='The readelf binary to run',
1155f9996aaSopenharmony_ci                        metavar='PATH')
1165f9996aaSopenharmony_ci    parser.add_argument('--nm',
1175f9996aaSopenharmony_ci                        required=True,
1185f9996aaSopenharmony_ci                        help='The nm binary to run',
1195f9996aaSopenharmony_ci                        metavar='PATH')
1205f9996aaSopenharmony_ci    parser.add_argument('--strip',
1215f9996aaSopenharmony_ci                        help='The strip binary to run',
1225f9996aaSopenharmony_ci                        metavar='PATH')
1235f9996aaSopenharmony_ci    parser.add_argument('--strip-debug-whitelist',
1245f9996aaSopenharmony_ci                        help='The strip debug whitelist, lines of which are names of shared objects with .symtab kept.',
1255f9996aaSopenharmony_ci                        metavar='PATH')
1265f9996aaSopenharmony_ci    parser.add_argument('--sofile',
1275f9996aaSopenharmony_ci                        required=True,
1285f9996aaSopenharmony_ci                        help='Shared object file produced by linking command',
1295f9996aaSopenharmony_ci                        metavar='FILE')
1305f9996aaSopenharmony_ci    parser.add_argument('--tocfile',
1315f9996aaSopenharmony_ci                        required=False,
1325f9996aaSopenharmony_ci                        help='Output table-of-contents file',
1335f9996aaSopenharmony_ci                        metavar='FILE')
1345f9996aaSopenharmony_ci    parser.add_argument('--map-file',
1355f9996aaSopenharmony_ci                        help=('Use --Wl,-Map to generate a map file. Will be '
1365f9996aaSopenharmony_ci                              'gzipped if extension ends with .gz'),
1375f9996aaSopenharmony_ci                        metavar='FILE')
1385f9996aaSopenharmony_ci    parser.add_argument('--output',
1395f9996aaSopenharmony_ci                        required=True,
1405f9996aaSopenharmony_ci                        help='Final output shared object file',
1415f9996aaSopenharmony_ci                        metavar='FILE')
1425f9996aaSopenharmony_ci    parser.add_argument('--libfile', required=False, metavar='FILE')
1435f9996aaSopenharmony_ci    parser.add_argument('command', nargs='+', help='Linking command')
1445f9996aaSopenharmony_ci    parser.add_argument('--mini-debug',
1455f9996aaSopenharmony_ci                        action='store_true',
1465f9996aaSopenharmony_ci                        default=False,
1475f9996aaSopenharmony_ci                        help='Add .gnu_debugdata section for stripped sofile')
1485f9996aaSopenharmony_ci    parser.add_argument('--target-name', help='')
1495f9996aaSopenharmony_ci    parser.add_argument('--target-out-dir', help='')
1505f9996aaSopenharmony_ci    parser.add_argument('--allowed-lib-list', help='')
1515f9996aaSopenharmony_ci    parser.add_argument('--clang-base-dir', help='')
1525f9996aaSopenharmony_ci    args = parser.parse_args()
1535f9996aaSopenharmony_ci
1545f9996aaSopenharmony_ci    if args.sofile.endswith(".dll"):
1555f9996aaSopenharmony_ci        rspfile = None
1565f9996aaSopenharmony_ci        for a in args.command:
1575f9996aaSopenharmony_ci            if a[0] == "@":
1585f9996aaSopenharmony_ci                rspfile = a[1:]
1595f9996aaSopenharmony_ci                break
1605f9996aaSopenharmony_ci        if rspfile:
1615f9996aaSopenharmony_ci            reformat_rsp_file(rspfile)
1625f9996aaSopenharmony_ci    # Work-around for gold being slow-by-default. http://crbug.com/632230
1635f9996aaSopenharmony_ci    fast_env = dict(os.environ)
1645f9996aaSopenharmony_ci    fast_env['LC_ALL'] = 'C'
1655f9996aaSopenharmony_ci
1665f9996aaSopenharmony_ci    # First, run the actual link.
1675f9996aaSopenharmony_ci    command = wrapper_utils.command_to_run(args.command)
1685f9996aaSopenharmony_ci    result = wrapper_utils.run_link_with_optional_map_file(
1695f9996aaSopenharmony_ci        command, env=fast_env, map_file=args.map_file)
1705f9996aaSopenharmony_ci
1715f9996aaSopenharmony_ci    if result != 0:
1725f9996aaSopenharmony_ci        return result
1735f9996aaSopenharmony_ci
1745f9996aaSopenharmony_ci    # Next, generate the contents of the TOC file.
1755f9996aaSopenharmony_ci    result, toc = collect_toc(args)
1765f9996aaSopenharmony_ci    if result != 0:
1775f9996aaSopenharmony_ci        return result
1785f9996aaSopenharmony_ci
1795f9996aaSopenharmony_ci    # If there is an existing TOC file with identical contents, leave it alone.
1805f9996aaSopenharmony_ci    # Otherwise, write out the TOC file.
1815f9996aaSopenharmony_ci    if args.tocfile:
1825f9996aaSopenharmony_ci        update_toc(args.tocfile, toc)
1835f9996aaSopenharmony_ci
1845f9996aaSopenharmony_ci    # Finally, strip the linked shared object file (if desired).
1855f9996aaSopenharmony_ci    if args.strip:
1865f9996aaSopenharmony_ci        strip_option = []
1875f9996aaSopenharmony_ci        strip_command = [args.strip, '-o', args.output]
1885f9996aaSopenharmony_ci
1895f9996aaSopenharmony_ci        #ADLT so should not be stripped
1905f9996aaSopenharmony_ci        if args.target_out_dir and os.path.exists(args.target_out_dir):
1915f9996aaSopenharmony_ci            install_dest = get_install_dest_dir(args)
1925f9996aaSopenharmony_ci        else:
1935f9996aaSopenharmony_ci            install_dest = None
1945f9996aaSopenharmony_ci        if install_dest:
1955f9996aaSopenharmony_ci            with open(args.allowed_lib_list, 'r') as f:
1965f9996aaSopenharmony_ci                lines = f.readlines()
1975f9996aaSopenharmony_ci            lines = [line.strip()[1:] for line in lines]
1985f9996aaSopenharmony_ci            if install_dest in lines:
1995f9996aaSopenharmony_ci                strip_option.extend(['-S'])
2005f9996aaSopenharmony_ci        if args.strip_debug_whitelist:
2015f9996aaSopenharmony_ci            with open(args.strip_debug_whitelist, 'r') as whitelist:
2025f9996aaSopenharmony_ci                for strip_debug_sofile in whitelist.readlines():
2035f9996aaSopenharmony_ci                    if args.sofile.endswith(strip_debug_sofile.strip()):
2045f9996aaSopenharmony_ci                        strip_option.extend(['--strip-debug', '-R', '.comment'])
2055f9996aaSopenharmony_ci                        break
2065f9996aaSopenharmony_ci        strip_command.extend(strip_option)
2075f9996aaSopenharmony_ci
2085f9996aaSopenharmony_ci        strip_command.append(args.sofile)
2095f9996aaSopenharmony_ci        result = subprocess.call(
2105f9996aaSopenharmony_ci            wrapper_utils.command_to_run(strip_command))
2115f9996aaSopenharmony_ci    if args.libfile:
2125f9996aaSopenharmony_ci        libfile_name = os.path.basename(args.libfile)
2135f9996aaSopenharmony_ci        sofile_output_dir = os.path.dirname(args.sofile)
2145f9996aaSopenharmony_ci        unstripped_libfile = os.path.join(sofile_output_dir, libfile_name)
2155f9996aaSopenharmony_ci        shutil.copy2(unstripped_libfile, args.libfile)
2165f9996aaSopenharmony_ci
2175f9996aaSopenharmony_ci    if args.mini_debug and not args.sofile.endswith(".dll"):
2185f9996aaSopenharmony_ci        unstripped_libfile = os.path.abspath(args.sofile)
2195f9996aaSopenharmony_ci        script_path = os.path.join(
2205f9996aaSopenharmony_ci            os.path.dirname(__file__), 'mini_debug_info.py')
2215f9996aaSopenharmony_ci        ohos_root_path = os.path.join(os.path.dirname(__file__), '../..')
2225f9996aaSopenharmony_ci        result = subprocess.call(
2235f9996aaSopenharmony_ci            wrapper_utils.command_to_run(
2245f9996aaSopenharmony_ci                ['python3', script_path, '--unstripped-path', unstripped_libfile, '--stripped-path', args.output,
2255f9996aaSopenharmony_ci                '--root-path', ohos_root_path, '--clang-base-dir', args.clang_base_dir]))
2265f9996aaSopenharmony_ci
2275f9996aaSopenharmony_ci    return result
2285f9996aaSopenharmony_ci
2295f9996aaSopenharmony_ci
2305f9996aaSopenharmony_ciif __name__ == "__main__":
2315f9996aaSopenharmony_ci    sys.exit(main())
232