| OLD | NEW |
| (Empty) |
| 1 #!/usr/bin/python | |
| 2 # | |
| 3 # libjingle | |
| 4 # Copyright 2015 Google Inc. | |
| 5 # | |
| 6 # Redistribution and use in source and binary forms, with or without | |
| 7 # modification, are permitted provided that the following conditions are met: | |
| 8 # | |
| 9 # 1. Redistributions of source code must retain the above copyright notice, | |
| 10 # this list of conditions and the following disclaimer. | |
| 11 # 2. Redistributions in binary form must reproduce the above copyright notice, | |
| 12 # this list of conditions and the following disclaimer in the documentation | |
| 13 # and/or other materials provided with the distribution. | |
| 14 # 3. The name of the author may not be used to endorse or promote products | |
| 15 # derived from this software without specific prior written permission. | |
| 16 # | |
| 17 # THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED | |
| 18 # WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF | |
| 19 # MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO | |
| 20 # EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, | |
| 21 # SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, | |
| 22 # PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; | |
| 23 # OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, | |
| 24 # WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR | |
| 25 # OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF | |
| 26 # ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | |
| 27 | |
| 28 """Script for merging generated iOS libraries.""" | |
| 29 | |
| 30 import optparse | |
| 31 import os | |
| 32 import re | |
| 33 import subprocess | |
| 34 import sys | |
| 35 | |
| 36 | |
| 37 def MergeLibs(lib_base_dir): | |
| 38 """Merges generated iOS libraries for different archs. | |
| 39 | |
| 40 Uses libtool to generate FAT archive files for each generated library. | |
| 41 | |
| 42 Args: | |
| 43 lib_base_dir: directory whose subdirectories are named by architecture and | |
| 44 contain the built libraries for that architecture | |
| 45 | |
| 46 Returns: | |
| 47 Exit code of libtool. | |
| 48 """ | |
| 49 include_dir_name = 'include' | |
| 50 output_dir_name = 'lib' | |
| 51 archs = [arch for arch in os.listdir(lib_base_dir) | |
| 52 if arch[:1] != '.' and arch != output_dir_name | |
| 53 and arch != include_dir_name] | |
| 54 # For each arch, find (library name, libary path) for arch. We will merge | |
| 55 # all libraries with the same name. | |
| 56 libs = {} | |
| 57 for dirpath, _, filenames in os.walk(lib_base_dir): | |
| 58 if dirpath.endswith(output_dir_name): | |
| 59 continue | |
| 60 for filename in filenames: | |
| 61 if not filename.endswith('.a'): | |
| 62 continue | |
| 63 entry = libs.get(filename, []) | |
| 64 entry.append(os.path.join(dirpath, filename)) | |
| 65 libs[filename] = entry | |
| 66 | |
| 67 orphaned_libs = {} | |
| 68 valid_libs = {} | |
| 69 for library, paths in libs.items(): | |
| 70 if len(paths) < len(archs): | |
| 71 orphaned_libs[library] = paths | |
| 72 else: | |
| 73 valid_libs[library] = paths | |
| 74 for library, paths in orphaned_libs.items(): | |
| 75 components = library[:-2].split('_')[:-1] | |
| 76 found = False | |
| 77 # Find directly matching parent libs by stripping suffix. | |
| 78 while components and not found: | |
| 79 parent_library = '_'.join(components) + '.a' | |
| 80 if parent_library in valid_libs: | |
| 81 valid_libs[parent_library].extend(paths) | |
| 82 found = True | |
| 83 break | |
| 84 components = components[:-1] | |
| 85 # Find next best match by finding parent libs with the same prefix. | |
| 86 if not found: | |
| 87 base_prefix = library[:-2].split('_')[0] | |
| 88 for valid_lib, valid_paths in valid_libs.items(): | |
| 89 prefix = '_'.join(components) | |
| 90 if valid_lib[:len(base_prefix)] == base_prefix: | |
| 91 valid_paths.extend(paths) | |
| 92 found = True | |
| 93 break | |
| 94 assert found | |
| 95 | |
| 96 # Create output directory. | |
| 97 output_dir_path = os.path.join(lib_base_dir, output_dir_name) | |
| 98 if not os.path.exists(output_dir_path): | |
| 99 os.mkdir(output_dir_path) | |
| 100 | |
| 101 # Use this so libtool merged binaries are always the same. | |
| 102 env = os.environ.copy() | |
| 103 env['ZERO_AR_DATE'] = '1' | |
| 104 | |
| 105 # Ignore certain errors. | |
| 106 libtool_re = re.compile(r'^.*libtool:.*file: .* has no symbols$') | |
| 107 | |
| 108 # Merge libraries using libtool. | |
| 109 for library, paths in valid_libs.items(): | |
| 110 cmd_list = ['libtool', '-static', '-v', '-o', | |
| 111 os.path.join(output_dir_path, library)] + paths | |
| 112 libtoolout = subprocess.Popen(cmd_list, stderr=subprocess.PIPE, env=env) | |
| 113 _, err = libtoolout.communicate() | |
| 114 for line in err.splitlines(): | |
| 115 if not libtool_re.match(line): | |
| 116 print >>sys.stderr, line | |
| 117 # Unconditionally touch the output .a file on the command line if present | |
| 118 # and the command succeeded. A bit hacky. | |
| 119 if not libtoolout.returncode: | |
| 120 for i in range(len(cmd_list) - 1): | |
| 121 if cmd_list[i] == '-o' and cmd_list[i+1].endswith('.a'): | |
| 122 os.utime(cmd_list[i+1], None) | |
| 123 break | |
| 124 else: | |
| 125 return libtoolout.returncode | |
| 126 return libtoolout.returncode | |
| 127 | |
| 128 | |
| 129 def Main(): | |
| 130 parser = optparse.OptionParser() | |
| 131 _, args = parser.parse_args() | |
| 132 if len(args) != 1: | |
| 133 parser.error('Error: Exactly 1 argument required.') | |
| 134 lib_base_dir = args[0] | |
| 135 MergeLibs(lib_base_dir) | |
| 136 | |
| 137 if __name__ == '__main__': | |
| 138 sys.exit(Main()) | |
| OLD | NEW |