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