xref: /aosp_15_r20/external/cronet/build/android/gyp/apkbuilder.py (revision 6777b5387eb2ff775bb5750e3f5d96f37fb7352b)
1*6777b538SAndroid Build Coastguard Worker#!/usr/bin/env python3
2*6777b538SAndroid Build Coastguard Worker#
3*6777b538SAndroid Build Coastguard Worker# Copyright 2015 The Chromium Authors
4*6777b538SAndroid Build Coastguard Worker# Use of this source code is governed by a BSD-style license that can be
5*6777b538SAndroid Build Coastguard Worker# found in the LICENSE file.
6*6777b538SAndroid Build Coastguard Worker
7*6777b538SAndroid Build Coastguard Worker"""Adds the code parts to a resource APK."""
8*6777b538SAndroid Build Coastguard Worker
9*6777b538SAndroid Build Coastguard Workerimport argparse
10*6777b538SAndroid Build Coastguard Workerimport logging
11*6777b538SAndroid Build Coastguard Workerimport os
12*6777b538SAndroid Build Coastguard Workerimport posixpath
13*6777b538SAndroid Build Coastguard Workerimport shutil
14*6777b538SAndroid Build Coastguard Workerimport sys
15*6777b538SAndroid Build Coastguard Workerimport tempfile
16*6777b538SAndroid Build Coastguard Workerimport zipfile
17*6777b538SAndroid Build Coastguard Workerimport zlib
18*6777b538SAndroid Build Coastguard Worker
19*6777b538SAndroid Build Coastguard Workerimport finalize_apk
20*6777b538SAndroid Build Coastguard Worker
21*6777b538SAndroid Build Coastguard Workerfrom util import build_utils
22*6777b538SAndroid Build Coastguard Workerfrom util import diff_utils
23*6777b538SAndroid Build Coastguard Workerimport action_helpers  # build_utils adds //build to sys.path.
24*6777b538SAndroid Build Coastguard Workerimport zip_helpers
25*6777b538SAndroid Build Coastguard Worker
26*6777b538SAndroid Build Coastguard Worker
27*6777b538SAndroid Build Coastguard Worker# Taken from aapt's Package.cpp:
28*6777b538SAndroid Build Coastguard Worker_NO_COMPRESS_EXTENSIONS = ('.jpg', '.jpeg', '.png', '.gif', '.wav', '.mp2',
29*6777b538SAndroid Build Coastguard Worker                           '.mp3', '.ogg', '.aac', '.mpg', '.mpeg', '.mid',
30*6777b538SAndroid Build Coastguard Worker                           '.midi', '.smf', '.jet', '.rtttl', '.imy', '.xmf',
31*6777b538SAndroid Build Coastguard Worker                           '.mp4', '.m4a', '.m4v', '.3gp', '.3gpp', '.3g2',
32*6777b538SAndroid Build Coastguard Worker                           '.3gpp2', '.amr', '.awb', '.wma', '.wmv', '.webm')
33*6777b538SAndroid Build Coastguard Worker
34*6777b538SAndroid Build Coastguard Worker
35*6777b538SAndroid Build Coastguard Workerdef _ParseArgs(args):
36*6777b538SAndroid Build Coastguard Worker  parser = argparse.ArgumentParser()
37*6777b538SAndroid Build Coastguard Worker  action_helpers.add_depfile_arg(parser)
38*6777b538SAndroid Build Coastguard Worker  parser.add_argument('--assets',
39*6777b538SAndroid Build Coastguard Worker                      action='append',
40*6777b538SAndroid Build Coastguard Worker                      help='GYP-list of files to add as assets in the form '
41*6777b538SAndroid Build Coastguard Worker                      '"srcPath:zipPath", where ":zipPath" is optional.')
42*6777b538SAndroid Build Coastguard Worker  parser.add_argument(
43*6777b538SAndroid Build Coastguard Worker      '--java-resources', help='GYP-list of java_resources JARs to include.')
44*6777b538SAndroid Build Coastguard Worker  parser.add_argument('--write-asset-list',
45*6777b538SAndroid Build Coastguard Worker                      action='store_true',
46*6777b538SAndroid Build Coastguard Worker                      help='Whether to create an assets/assets_list file.')
47*6777b538SAndroid Build Coastguard Worker  parser.add_argument(
48*6777b538SAndroid Build Coastguard Worker      '--uncompressed-assets',
49*6777b538SAndroid Build Coastguard Worker      help='Same as --assets, except disables compression.')
50*6777b538SAndroid Build Coastguard Worker  parser.add_argument('--resource-apk',
51*6777b538SAndroid Build Coastguard Worker                      help='An .ap_ file built using aapt',
52*6777b538SAndroid Build Coastguard Worker                      required=True)
53*6777b538SAndroid Build Coastguard Worker  parser.add_argument('--output-apk',
54*6777b538SAndroid Build Coastguard Worker                      help='Path to the output file',
55*6777b538SAndroid Build Coastguard Worker                      required=True)
56*6777b538SAndroid Build Coastguard Worker  parser.add_argument('--format', choices=['apk', 'bundle-module'],
57*6777b538SAndroid Build Coastguard Worker                      default='apk', help='Specify output format.')
58*6777b538SAndroid Build Coastguard Worker  parser.add_argument('--dex-file',
59*6777b538SAndroid Build Coastguard Worker                      help='Path to the classes.dex to use')
60*6777b538SAndroid Build Coastguard Worker  parser.add_argument('--uncompress-dex', action='store_true',
61*6777b538SAndroid Build Coastguard Worker                      help='Store .dex files uncompressed in the APK')
62*6777b538SAndroid Build Coastguard Worker  parser.add_argument('--native-libs',
63*6777b538SAndroid Build Coastguard Worker                      action='append',
64*6777b538SAndroid Build Coastguard Worker                      help='GYP-list of native libraries to include. '
65*6777b538SAndroid Build Coastguard Worker                           'Can be specified multiple times.',
66*6777b538SAndroid Build Coastguard Worker                      default=[])
67*6777b538SAndroid Build Coastguard Worker  parser.add_argument('--secondary-native-libs',
68*6777b538SAndroid Build Coastguard Worker                      action='append',
69*6777b538SAndroid Build Coastguard Worker                      help='GYP-list of native libraries for secondary '
70*6777b538SAndroid Build Coastguard Worker                           'android-abi. Can be specified multiple times.',
71*6777b538SAndroid Build Coastguard Worker                      default=[])
72*6777b538SAndroid Build Coastguard Worker  parser.add_argument('--android-abi',
73*6777b538SAndroid Build Coastguard Worker                      help='Android architecture to use for native libraries')
74*6777b538SAndroid Build Coastguard Worker  parser.add_argument('--secondary-android-abi',
75*6777b538SAndroid Build Coastguard Worker                      help='The secondary Android architecture to use for'
76*6777b538SAndroid Build Coastguard Worker                           'secondary native libraries')
77*6777b538SAndroid Build Coastguard Worker  parser.add_argument(
78*6777b538SAndroid Build Coastguard Worker      '--is-multi-abi',
79*6777b538SAndroid Build Coastguard Worker      action='store_true',
80*6777b538SAndroid Build Coastguard Worker      help='Will add a placeholder for the missing ABI if no native libs or '
81*6777b538SAndroid Build Coastguard Worker      'placeholders are set for either the primary or secondary ABI. Can only '
82*6777b538SAndroid Build Coastguard Worker      'be set if both --android-abi and --secondary-android-abi are set.')
83*6777b538SAndroid Build Coastguard Worker  parser.add_argument(
84*6777b538SAndroid Build Coastguard Worker      '--native-lib-placeholders',
85*6777b538SAndroid Build Coastguard Worker      help='GYP-list of native library placeholders to add.')
86*6777b538SAndroid Build Coastguard Worker  parser.add_argument(
87*6777b538SAndroid Build Coastguard Worker      '--secondary-native-lib-placeholders',
88*6777b538SAndroid Build Coastguard Worker      help='GYP-list of native library placeholders to add '
89*6777b538SAndroid Build Coastguard Worker      'for the secondary ABI')
90*6777b538SAndroid Build Coastguard Worker  parser.add_argument('--uncompress-shared-libraries', default='False',
91*6777b538SAndroid Build Coastguard Worker      choices=['true', 'True', 'false', 'False'],
92*6777b538SAndroid Build Coastguard Worker      help='Whether to uncompress native shared libraries. Argument must be '
93*6777b538SAndroid Build Coastguard Worker           'a boolean value.')
94*6777b538SAndroid Build Coastguard Worker  parser.add_argument(
95*6777b538SAndroid Build Coastguard Worker      '--apksigner-jar', help='Path to the apksigner executable.')
96*6777b538SAndroid Build Coastguard Worker  parser.add_argument('--zipalign-path',
97*6777b538SAndroid Build Coastguard Worker                      help='Path to the zipalign executable.')
98*6777b538SAndroid Build Coastguard Worker  parser.add_argument('--key-path',
99*6777b538SAndroid Build Coastguard Worker                      help='Path to keystore for signing.')
100*6777b538SAndroid Build Coastguard Worker  parser.add_argument('--key-passwd',
101*6777b538SAndroid Build Coastguard Worker                      help='Keystore password')
102*6777b538SAndroid Build Coastguard Worker  parser.add_argument('--key-name',
103*6777b538SAndroid Build Coastguard Worker                      help='Keystore name')
104*6777b538SAndroid Build Coastguard Worker  parser.add_argument(
105*6777b538SAndroid Build Coastguard Worker      '--min-sdk-version', required=True, help='Value of APK\'s minSdkVersion')
106*6777b538SAndroid Build Coastguard Worker  parser.add_argument(
107*6777b538SAndroid Build Coastguard Worker      '--best-compression',
108*6777b538SAndroid Build Coastguard Worker      action='store_true',
109*6777b538SAndroid Build Coastguard Worker      help='Use zip -9 rather than zip -1')
110*6777b538SAndroid Build Coastguard Worker  parser.add_argument(
111*6777b538SAndroid Build Coastguard Worker      '--library-always-compress',
112*6777b538SAndroid Build Coastguard Worker      action='append',
113*6777b538SAndroid Build Coastguard Worker      help='The list of library files that we always compress.')
114*6777b538SAndroid Build Coastguard Worker  parser.add_argument('--warnings-as-errors',
115*6777b538SAndroid Build Coastguard Worker                      action='store_true',
116*6777b538SAndroid Build Coastguard Worker                      help='Treat all warnings as errors.')
117*6777b538SAndroid Build Coastguard Worker  diff_utils.AddCommandLineFlags(parser)
118*6777b538SAndroid Build Coastguard Worker  options = parser.parse_args(args)
119*6777b538SAndroid Build Coastguard Worker  options.assets = action_helpers.parse_gn_list(options.assets)
120*6777b538SAndroid Build Coastguard Worker  options.uncompressed_assets = action_helpers.parse_gn_list(
121*6777b538SAndroid Build Coastguard Worker      options.uncompressed_assets)
122*6777b538SAndroid Build Coastguard Worker  options.native_lib_placeholders = action_helpers.parse_gn_list(
123*6777b538SAndroid Build Coastguard Worker      options.native_lib_placeholders)
124*6777b538SAndroid Build Coastguard Worker  options.secondary_native_lib_placeholders = action_helpers.parse_gn_list(
125*6777b538SAndroid Build Coastguard Worker      options.secondary_native_lib_placeholders)
126*6777b538SAndroid Build Coastguard Worker  options.java_resources = action_helpers.parse_gn_list(options.java_resources)
127*6777b538SAndroid Build Coastguard Worker  options.native_libs = action_helpers.parse_gn_list(options.native_libs)
128*6777b538SAndroid Build Coastguard Worker  options.secondary_native_libs = action_helpers.parse_gn_list(
129*6777b538SAndroid Build Coastguard Worker      options.secondary_native_libs)
130*6777b538SAndroid Build Coastguard Worker  options.library_always_compress = action_helpers.parse_gn_list(
131*6777b538SAndroid Build Coastguard Worker      options.library_always_compress)
132*6777b538SAndroid Build Coastguard Worker
133*6777b538SAndroid Build Coastguard Worker  if not options.android_abi and (options.native_libs or
134*6777b538SAndroid Build Coastguard Worker                                  options.native_lib_placeholders):
135*6777b538SAndroid Build Coastguard Worker    raise Exception('Must specify --android-abi with --native-libs')
136*6777b538SAndroid Build Coastguard Worker  if not options.secondary_android_abi and (options.secondary_native_libs or
137*6777b538SAndroid Build Coastguard Worker      options.secondary_native_lib_placeholders):
138*6777b538SAndroid Build Coastguard Worker    raise Exception('Must specify --secondary-android-abi with'
139*6777b538SAndroid Build Coastguard Worker                    ' --secondary-native-libs')
140*6777b538SAndroid Build Coastguard Worker  if options.is_multi_abi and not (options.android_abi
141*6777b538SAndroid Build Coastguard Worker                                   and options.secondary_android_abi):
142*6777b538SAndroid Build Coastguard Worker    raise Exception('Must specify --is-multi-abi with both --android-abi '
143*6777b538SAndroid Build Coastguard Worker                    'and --secondary-android-abi.')
144*6777b538SAndroid Build Coastguard Worker  return options
145*6777b538SAndroid Build Coastguard Worker
146*6777b538SAndroid Build Coastguard Worker
147*6777b538SAndroid Build Coastguard Workerdef _SplitAssetPath(path):
148*6777b538SAndroid Build Coastguard Worker  """Returns (src, dest) given an asset path in the form src[:dest]."""
149*6777b538SAndroid Build Coastguard Worker  path_parts = path.split(':')
150*6777b538SAndroid Build Coastguard Worker  src_path = path_parts[0]
151*6777b538SAndroid Build Coastguard Worker  if len(path_parts) > 1:
152*6777b538SAndroid Build Coastguard Worker    dest_path = path_parts[1]
153*6777b538SAndroid Build Coastguard Worker  else:
154*6777b538SAndroid Build Coastguard Worker    dest_path = os.path.basename(src_path)
155*6777b538SAndroid Build Coastguard Worker  return src_path, dest_path
156*6777b538SAndroid Build Coastguard Worker
157*6777b538SAndroid Build Coastguard Worker
158*6777b538SAndroid Build Coastguard Workerdef _ExpandPaths(paths):
159*6777b538SAndroid Build Coastguard Worker  """Converts src:dst into tuples and enumerates files within directories.
160*6777b538SAndroid Build Coastguard Worker
161*6777b538SAndroid Build Coastguard Worker  Args:
162*6777b538SAndroid Build Coastguard Worker    paths: Paths in the form "src_path:dest_path"
163*6777b538SAndroid Build Coastguard Worker
164*6777b538SAndroid Build Coastguard Worker  Returns:
165*6777b538SAndroid Build Coastguard Worker    A list of (src_path, dest_path) tuples sorted by dest_path (for stable
166*6777b538SAndroid Build Coastguard Worker    ordering within output .apk).
167*6777b538SAndroid Build Coastguard Worker  """
168*6777b538SAndroid Build Coastguard Worker  ret = []
169*6777b538SAndroid Build Coastguard Worker  for path in paths:
170*6777b538SAndroid Build Coastguard Worker    src_path, dest_path = _SplitAssetPath(path)
171*6777b538SAndroid Build Coastguard Worker    if os.path.isdir(src_path):
172*6777b538SAndroid Build Coastguard Worker      for f in build_utils.FindInDirectory(src_path, '*'):
173*6777b538SAndroid Build Coastguard Worker        ret.append((f, os.path.join(dest_path, f[len(src_path) + 1:])))
174*6777b538SAndroid Build Coastguard Worker    else:
175*6777b538SAndroid Build Coastguard Worker      ret.append((src_path, dest_path))
176*6777b538SAndroid Build Coastguard Worker  ret.sort(key=lambda t:t[1])
177*6777b538SAndroid Build Coastguard Worker  return ret
178*6777b538SAndroid Build Coastguard Worker
179*6777b538SAndroid Build Coastguard Worker
180*6777b538SAndroid Build Coastguard Workerdef _GetAssetsToAdd(path_tuples,
181*6777b538SAndroid Build Coastguard Worker                    fast_align,
182*6777b538SAndroid Build Coastguard Worker                    disable_compression=False,
183*6777b538SAndroid Build Coastguard Worker                    allow_reads=True,
184*6777b538SAndroid Build Coastguard Worker                    apk_root_dir=''):
185*6777b538SAndroid Build Coastguard Worker  """Returns the list of file_detail tuples for assets in the apk.
186*6777b538SAndroid Build Coastguard Worker
187*6777b538SAndroid Build Coastguard Worker  Args:
188*6777b538SAndroid Build Coastguard Worker    path_tuples: List of src_path, dest_path tuples to add.
189*6777b538SAndroid Build Coastguard Worker    fast_align: Whether to perform alignment in python zipfile (alternatively
190*6777b538SAndroid Build Coastguard Worker                alignment can be done using the zipalign utility out of band).
191*6777b538SAndroid Build Coastguard Worker    disable_compression: Whether to disable compression.
192*6777b538SAndroid Build Coastguard Worker    allow_reads: If false, we do not try to read the files from disk (to find
193*6777b538SAndroid Build Coastguard Worker                 their size for example).
194*6777b538SAndroid Build Coastguard Worker
195*6777b538SAndroid Build Coastguard Worker  Returns: A list of (src_path, apk_path, compress, alignment) tuple
196*6777b538SAndroid Build Coastguard Worker  representing what and how assets are added.
197*6777b538SAndroid Build Coastguard Worker  """
198*6777b538SAndroid Build Coastguard Worker  assets_to_add = []
199*6777b538SAndroid Build Coastguard Worker
200*6777b538SAndroid Build Coastguard Worker  # Group all uncompressed assets together in the hope that it will increase
201*6777b538SAndroid Build Coastguard Worker  # locality of mmap'ed files.
202*6777b538SAndroid Build Coastguard Worker  for target_compress in (False, True):
203*6777b538SAndroid Build Coastguard Worker    for src_path, dest_path in path_tuples:
204*6777b538SAndroid Build Coastguard Worker      compress = not disable_compression and (
205*6777b538SAndroid Build Coastguard Worker          os.path.splitext(src_path)[1] not in _NO_COMPRESS_EXTENSIONS)
206*6777b538SAndroid Build Coastguard Worker
207*6777b538SAndroid Build Coastguard Worker      if target_compress == compress:
208*6777b538SAndroid Build Coastguard Worker        # add_to_zip_hermetic() uses this logic to avoid growing small files.
209*6777b538SAndroid Build Coastguard Worker        # We need it here in order to set alignment correctly.
210*6777b538SAndroid Build Coastguard Worker        if allow_reads and compress and os.path.getsize(src_path) < 16:
211*6777b538SAndroid Build Coastguard Worker          compress = False
212*6777b538SAndroid Build Coastguard Worker
213*6777b538SAndroid Build Coastguard Worker        if dest_path.startswith('../'):
214*6777b538SAndroid Build Coastguard Worker          # posixpath.join('', 'foo') == 'foo'
215*6777b538SAndroid Build Coastguard Worker          apk_path = posixpath.join(apk_root_dir, dest_path[3:])
216*6777b538SAndroid Build Coastguard Worker        else:
217*6777b538SAndroid Build Coastguard Worker          apk_path = 'assets/' + dest_path
218*6777b538SAndroid Build Coastguard Worker        alignment = 0 if compress and not fast_align else 4
219*6777b538SAndroid Build Coastguard Worker        assets_to_add.append((apk_path, src_path, compress, alignment))
220*6777b538SAndroid Build Coastguard Worker  return assets_to_add
221*6777b538SAndroid Build Coastguard Worker
222*6777b538SAndroid Build Coastguard Worker
223*6777b538SAndroid Build Coastguard Workerdef _AddFiles(apk, details):
224*6777b538SAndroid Build Coastguard Worker  """Adds files to the apk.
225*6777b538SAndroid Build Coastguard Worker
226*6777b538SAndroid Build Coastguard Worker  Args:
227*6777b538SAndroid Build Coastguard Worker    apk: path to APK to add to.
228*6777b538SAndroid Build Coastguard Worker    details: A list of file detail tuples (src_path, apk_path, compress,
229*6777b538SAndroid Build Coastguard Worker    alignment) representing what and how files are added to the APK.
230*6777b538SAndroid Build Coastguard Worker  """
231*6777b538SAndroid Build Coastguard Worker  for apk_path, src_path, compress, alignment in details:
232*6777b538SAndroid Build Coastguard Worker    # This check is only relevant for assets, but it should not matter if it is
233*6777b538SAndroid Build Coastguard Worker    # checked for the whole list of files.
234*6777b538SAndroid Build Coastguard Worker    try:
235*6777b538SAndroid Build Coastguard Worker      apk.getinfo(apk_path)
236*6777b538SAndroid Build Coastguard Worker      # Should never happen since write_build_config.py handles merging.
237*6777b538SAndroid Build Coastguard Worker      raise Exception(
238*6777b538SAndroid Build Coastguard Worker          'Multiple targets specified the asset path: %s' % apk_path)
239*6777b538SAndroid Build Coastguard Worker    except KeyError:
240*6777b538SAndroid Build Coastguard Worker      zip_helpers.add_to_zip_hermetic(apk,
241*6777b538SAndroid Build Coastguard Worker                                      apk_path,
242*6777b538SAndroid Build Coastguard Worker                                      src_path=src_path,
243*6777b538SAndroid Build Coastguard Worker                                      compress=compress,
244*6777b538SAndroid Build Coastguard Worker                                      alignment=alignment)
245*6777b538SAndroid Build Coastguard Worker
246*6777b538SAndroid Build Coastguard Worker
247*6777b538SAndroid Build Coastguard Workerdef _GetAbiAlignment(android_abi):
248*6777b538SAndroid Build Coastguard Worker  if '64' in android_abi:
249*6777b538SAndroid Build Coastguard Worker    return 0x4000  # 16k alignment
250*6777b538SAndroid Build Coastguard Worker  return 0x1000  # 4k alignment
251*6777b538SAndroid Build Coastguard Worker
252*6777b538SAndroid Build Coastguard Worker
253*6777b538SAndroid Build Coastguard Workerdef _GetNativeLibrariesToAdd(native_libs, android_abi, fast_align,
254*6777b538SAndroid Build Coastguard Worker                             lib_always_compress):
255*6777b538SAndroid Build Coastguard Worker  """Returns the list of file_detail tuples for native libraries in the apk.
256*6777b538SAndroid Build Coastguard Worker
257*6777b538SAndroid Build Coastguard Worker  Returns: A list of (src_path, apk_path, compress, alignment) tuple
258*6777b538SAndroid Build Coastguard Worker  representing what and how native libraries are added.
259*6777b538SAndroid Build Coastguard Worker  """
260*6777b538SAndroid Build Coastguard Worker  libraries_to_add = []
261*6777b538SAndroid Build Coastguard Worker
262*6777b538SAndroid Build Coastguard Worker
263*6777b538SAndroid Build Coastguard Worker  for path in native_libs:
264*6777b538SAndroid Build Coastguard Worker    basename = os.path.basename(path)
265*6777b538SAndroid Build Coastguard Worker    compress = any(lib_name in basename for lib_name in lib_always_compress)
266*6777b538SAndroid Build Coastguard Worker    lib_android_abi = android_abi
267*6777b538SAndroid Build Coastguard Worker    if path.startswith('android_clang_arm64_hwasan/'):
268*6777b538SAndroid Build Coastguard Worker      lib_android_abi = 'arm64-v8a-hwasan'
269*6777b538SAndroid Build Coastguard Worker
270*6777b538SAndroid Build Coastguard Worker    apk_path = 'lib/%s/%s' % (lib_android_abi, basename)
271*6777b538SAndroid Build Coastguard Worker    if compress and not fast_align:
272*6777b538SAndroid Build Coastguard Worker      alignment = 0
273*6777b538SAndroid Build Coastguard Worker    else:
274*6777b538SAndroid Build Coastguard Worker      alignment = _GetAbiAlignment(android_abi)
275*6777b538SAndroid Build Coastguard Worker    libraries_to_add.append((apk_path, path, compress, alignment))
276*6777b538SAndroid Build Coastguard Worker
277*6777b538SAndroid Build Coastguard Worker  return libraries_to_add
278*6777b538SAndroid Build Coastguard Worker
279*6777b538SAndroid Build Coastguard Worker
280*6777b538SAndroid Build Coastguard Workerdef _CreateExpectationsData(native_libs, assets):
281*6777b538SAndroid Build Coastguard Worker  """Creates list of native libraries and assets."""
282*6777b538SAndroid Build Coastguard Worker  native_libs = sorted(native_libs)
283*6777b538SAndroid Build Coastguard Worker  assets = sorted(assets)
284*6777b538SAndroid Build Coastguard Worker
285*6777b538SAndroid Build Coastguard Worker  ret = []
286*6777b538SAndroid Build Coastguard Worker  for apk_path, _, compress, alignment in native_libs + assets:
287*6777b538SAndroid Build Coastguard Worker    ret.append('apk_path=%s, compress=%s, alignment=%s\n' %
288*6777b538SAndroid Build Coastguard Worker               (apk_path, compress, alignment))
289*6777b538SAndroid Build Coastguard Worker  return ''.join(ret)
290*6777b538SAndroid Build Coastguard Worker
291*6777b538SAndroid Build Coastguard Worker
292*6777b538SAndroid Build Coastguard Workerdef main(args):
293*6777b538SAndroid Build Coastguard Worker  build_utils.InitLogging('APKBUILDER_DEBUG')
294*6777b538SAndroid Build Coastguard Worker  args = build_utils.ExpandFileArgs(args)
295*6777b538SAndroid Build Coastguard Worker  options = _ParseArgs(args)
296*6777b538SAndroid Build Coastguard Worker
297*6777b538SAndroid Build Coastguard Worker  # Until Python 3.7, there's no better way to set compression level.
298*6777b538SAndroid Build Coastguard Worker  # The default is 6.
299*6777b538SAndroid Build Coastguard Worker  if options.best_compression:
300*6777b538SAndroid Build Coastguard Worker    # Compresses about twice as slow as the default.
301*6777b538SAndroid Build Coastguard Worker    zlib.Z_DEFAULT_COMPRESSION = 9
302*6777b538SAndroid Build Coastguard Worker  else:
303*6777b538SAndroid Build Coastguard Worker    # Compresses about twice as fast as the default.
304*6777b538SAndroid Build Coastguard Worker    zlib.Z_DEFAULT_COMPRESSION = 1
305*6777b538SAndroid Build Coastguard Worker
306*6777b538SAndroid Build Coastguard Worker  # Python's zip implementation duplicates file comments in the central
307*6777b538SAndroid Build Coastguard Worker  # directory, whereas zipalign does not, so use zipalign for official builds.
308*6777b538SAndroid Build Coastguard Worker  requires_alignment = options.format == 'apk'
309*6777b538SAndroid Build Coastguard Worker  # TODO(crbug.com/1495851): Re-enable zipalign once we are using Android V SDK.
310*6777b538SAndroid Build Coastguard Worker  run_zipalign = requires_alignment and options.best_compression and False
311*6777b538SAndroid Build Coastguard Worker  fast_align = bool(requires_alignment and not run_zipalign)
312*6777b538SAndroid Build Coastguard Worker
313*6777b538SAndroid Build Coastguard Worker  native_libs = sorted(options.native_libs)
314*6777b538SAndroid Build Coastguard Worker
315*6777b538SAndroid Build Coastguard Worker  # Include native libs in the depfile_deps since GN doesn't know about the
316*6777b538SAndroid Build Coastguard Worker  # dependencies when is_component_build=true.
317*6777b538SAndroid Build Coastguard Worker  depfile_deps = list(native_libs)
318*6777b538SAndroid Build Coastguard Worker
319*6777b538SAndroid Build Coastguard Worker  # For targets that depend on static library APKs, dex paths are created by
320*6777b538SAndroid Build Coastguard Worker  # the static library's dexsplitter target and GN doesn't know about these
321*6777b538SAndroid Build Coastguard Worker  # paths.
322*6777b538SAndroid Build Coastguard Worker  if options.dex_file:
323*6777b538SAndroid Build Coastguard Worker    depfile_deps.append(options.dex_file)
324*6777b538SAndroid Build Coastguard Worker
325*6777b538SAndroid Build Coastguard Worker  secondary_native_libs = []
326*6777b538SAndroid Build Coastguard Worker  if options.secondary_native_libs:
327*6777b538SAndroid Build Coastguard Worker    secondary_native_libs = sorted(options.secondary_native_libs)
328*6777b538SAndroid Build Coastguard Worker    depfile_deps += secondary_native_libs
329*6777b538SAndroid Build Coastguard Worker
330*6777b538SAndroid Build Coastguard Worker  if options.java_resources:
331*6777b538SAndroid Build Coastguard Worker    # Included via .build_config.json, so need to write it to depfile.
332*6777b538SAndroid Build Coastguard Worker    depfile_deps.extend(options.java_resources)
333*6777b538SAndroid Build Coastguard Worker
334*6777b538SAndroid Build Coastguard Worker  assets = _ExpandPaths(options.assets)
335*6777b538SAndroid Build Coastguard Worker  uncompressed_assets = _ExpandPaths(options.uncompressed_assets)
336*6777b538SAndroid Build Coastguard Worker
337*6777b538SAndroid Build Coastguard Worker  # Included via .build_config.json, so need to write it to depfile.
338*6777b538SAndroid Build Coastguard Worker  depfile_deps.extend(x[0] for x in assets)
339*6777b538SAndroid Build Coastguard Worker  depfile_deps.extend(x[0] for x in uncompressed_assets)
340*6777b538SAndroid Build Coastguard Worker  depfile_deps.append(options.resource_apk)
341*6777b538SAndroid Build Coastguard Worker
342*6777b538SAndroid Build Coastguard Worker  # Bundle modules have a structure similar to APKs, except that resources
343*6777b538SAndroid Build Coastguard Worker  # are compiled in protobuf format (instead of binary xml), and that some
344*6777b538SAndroid Build Coastguard Worker  # files are located into different top-level directories, e.g.:
345*6777b538SAndroid Build Coastguard Worker  #  AndroidManifest.xml -> manifest/AndroidManifest.xml
346*6777b538SAndroid Build Coastguard Worker  #  classes.dex -> dex/classes.dex
347*6777b538SAndroid Build Coastguard Worker  #  res/ -> res/  (unchanged)
348*6777b538SAndroid Build Coastguard Worker  #  assets/ -> assets/  (unchanged)
349*6777b538SAndroid Build Coastguard Worker  #  <other-file> -> root/<other-file>
350*6777b538SAndroid Build Coastguard Worker  #
351*6777b538SAndroid Build Coastguard Worker  # Hence, the following variables are used to control the location of files in
352*6777b538SAndroid Build Coastguard Worker  # the final archive.
353*6777b538SAndroid Build Coastguard Worker  if options.format == 'bundle-module':
354*6777b538SAndroid Build Coastguard Worker    apk_manifest_dir = 'manifest/'
355*6777b538SAndroid Build Coastguard Worker    apk_root_dir = 'root/'
356*6777b538SAndroid Build Coastguard Worker    apk_dex_dir = 'dex/'
357*6777b538SAndroid Build Coastguard Worker  else:
358*6777b538SAndroid Build Coastguard Worker    apk_manifest_dir = ''
359*6777b538SAndroid Build Coastguard Worker    apk_root_dir = ''
360*6777b538SAndroid Build Coastguard Worker    apk_dex_dir = ''
361*6777b538SAndroid Build Coastguard Worker
362*6777b538SAndroid Build Coastguard Worker  def _GetAssetDetails(assets, uncompressed_assets, fast_align, allow_reads):
363*6777b538SAndroid Build Coastguard Worker    ret = _GetAssetsToAdd(assets,
364*6777b538SAndroid Build Coastguard Worker                          fast_align,
365*6777b538SAndroid Build Coastguard Worker                          disable_compression=False,
366*6777b538SAndroid Build Coastguard Worker                          allow_reads=allow_reads,
367*6777b538SAndroid Build Coastguard Worker                          apk_root_dir=apk_root_dir)
368*6777b538SAndroid Build Coastguard Worker    ret.extend(
369*6777b538SAndroid Build Coastguard Worker        _GetAssetsToAdd(uncompressed_assets,
370*6777b538SAndroid Build Coastguard Worker                        fast_align,
371*6777b538SAndroid Build Coastguard Worker                        disable_compression=True,
372*6777b538SAndroid Build Coastguard Worker                        allow_reads=allow_reads,
373*6777b538SAndroid Build Coastguard Worker                        apk_root_dir=apk_root_dir))
374*6777b538SAndroid Build Coastguard Worker    return ret
375*6777b538SAndroid Build Coastguard Worker
376*6777b538SAndroid Build Coastguard Worker  libs_to_add = _GetNativeLibrariesToAdd(native_libs, options.android_abi,
377*6777b538SAndroid Build Coastguard Worker                                         fast_align,
378*6777b538SAndroid Build Coastguard Worker                                         options.library_always_compress)
379*6777b538SAndroid Build Coastguard Worker  if options.secondary_android_abi:
380*6777b538SAndroid Build Coastguard Worker    libs_to_add.extend(
381*6777b538SAndroid Build Coastguard Worker        _GetNativeLibrariesToAdd(secondary_native_libs,
382*6777b538SAndroid Build Coastguard Worker                                 options.secondary_android_abi,
383*6777b538SAndroid Build Coastguard Worker                                 fast_align, options.library_always_compress))
384*6777b538SAndroid Build Coastguard Worker
385*6777b538SAndroid Build Coastguard Worker  if options.expected_file:
386*6777b538SAndroid Build Coastguard Worker    # We compute expectations without reading the files. This allows us to check
387*6777b538SAndroid Build Coastguard Worker    # expectations for different targets by just generating their build_configs
388*6777b538SAndroid Build Coastguard Worker    # and not have to first generate all the actual files and all their
389*6777b538SAndroid Build Coastguard Worker    # dependencies (for example by just passing --only-verify-expectations).
390*6777b538SAndroid Build Coastguard Worker    asset_details = _GetAssetDetails(assets,
391*6777b538SAndroid Build Coastguard Worker                                     uncompressed_assets,
392*6777b538SAndroid Build Coastguard Worker                                     fast_align,
393*6777b538SAndroid Build Coastguard Worker                                     allow_reads=False)
394*6777b538SAndroid Build Coastguard Worker
395*6777b538SAndroid Build Coastguard Worker    actual_data = _CreateExpectationsData(libs_to_add, asset_details)
396*6777b538SAndroid Build Coastguard Worker    diff_utils.CheckExpectations(actual_data, options)
397*6777b538SAndroid Build Coastguard Worker
398*6777b538SAndroid Build Coastguard Worker    if options.only_verify_expectations:
399*6777b538SAndroid Build Coastguard Worker      if options.depfile:
400*6777b538SAndroid Build Coastguard Worker        action_helpers.write_depfile(options.depfile,
401*6777b538SAndroid Build Coastguard Worker                                     options.actual_file,
402*6777b538SAndroid Build Coastguard Worker                                     inputs=depfile_deps)
403*6777b538SAndroid Build Coastguard Worker      return
404*6777b538SAndroid Build Coastguard Worker
405*6777b538SAndroid Build Coastguard Worker  # If we are past this point, we are going to actually create the final apk so
406*6777b538SAndroid Build Coastguard Worker  # we should recompute asset details again but maybe perform some optimizations
407*6777b538SAndroid Build Coastguard Worker  # based on the size of the files on disk.
408*6777b538SAndroid Build Coastguard Worker  assets_to_add = _GetAssetDetails(
409*6777b538SAndroid Build Coastguard Worker      assets, uncompressed_assets, fast_align, allow_reads=True)
410*6777b538SAndroid Build Coastguard Worker
411*6777b538SAndroid Build Coastguard Worker  # Targets generally do not depend on apks, so no need for only_if_changed.
412*6777b538SAndroid Build Coastguard Worker  with action_helpers.atomic_output(options.output_apk,
413*6777b538SAndroid Build Coastguard Worker                                    only_if_changed=False) as f:
414*6777b538SAndroid Build Coastguard Worker    with zipfile.ZipFile(options.resource_apk) as resource_apk, \
415*6777b538SAndroid Build Coastguard Worker         zipfile.ZipFile(f, 'w') as out_apk:
416*6777b538SAndroid Build Coastguard Worker
417*6777b538SAndroid Build Coastguard Worker      def add_to_zip(zip_path, data, compress=True, alignment=4):
418*6777b538SAndroid Build Coastguard Worker        zip_helpers.add_to_zip_hermetic(
419*6777b538SAndroid Build Coastguard Worker            out_apk,
420*6777b538SAndroid Build Coastguard Worker            zip_path,
421*6777b538SAndroid Build Coastguard Worker            data=data,
422*6777b538SAndroid Build Coastguard Worker            compress=compress,
423*6777b538SAndroid Build Coastguard Worker            alignment=0 if compress and not fast_align else alignment)
424*6777b538SAndroid Build Coastguard Worker
425*6777b538SAndroid Build Coastguard Worker      def copy_resource(zipinfo, out_dir=''):
426*6777b538SAndroid Build Coastguard Worker        add_to_zip(
427*6777b538SAndroid Build Coastguard Worker            out_dir + zipinfo.filename,
428*6777b538SAndroid Build Coastguard Worker            resource_apk.read(zipinfo.filename),
429*6777b538SAndroid Build Coastguard Worker            compress=zipinfo.compress_type != zipfile.ZIP_STORED)
430*6777b538SAndroid Build Coastguard Worker
431*6777b538SAndroid Build Coastguard Worker      # Make assets come before resources in order to maintain the same file
432*6777b538SAndroid Build Coastguard Worker      # ordering as GYP / aapt. http://crbug.com/561862
433*6777b538SAndroid Build Coastguard Worker      resource_infos = resource_apk.infolist()
434*6777b538SAndroid Build Coastguard Worker
435*6777b538SAndroid Build Coastguard Worker      # 1. AndroidManifest.xml
436*6777b538SAndroid Build Coastguard Worker      logging.debug('Adding AndroidManifest.xml')
437*6777b538SAndroid Build Coastguard Worker      copy_resource(
438*6777b538SAndroid Build Coastguard Worker          resource_apk.getinfo('AndroidManifest.xml'), out_dir=apk_manifest_dir)
439*6777b538SAndroid Build Coastguard Worker
440*6777b538SAndroid Build Coastguard Worker      # 2. Assets
441*6777b538SAndroid Build Coastguard Worker      logging.debug('Adding assets/')
442*6777b538SAndroid Build Coastguard Worker      _AddFiles(out_apk, assets_to_add)
443*6777b538SAndroid Build Coastguard Worker
444*6777b538SAndroid Build Coastguard Worker      # 3. Dex files
445*6777b538SAndroid Build Coastguard Worker      logging.debug('Adding classes.dex')
446*6777b538SAndroid Build Coastguard Worker      if options.dex_file:
447*6777b538SAndroid Build Coastguard Worker        with open(options.dex_file, 'rb') as dex_file_obj:
448*6777b538SAndroid Build Coastguard Worker          if options.dex_file.endswith('.dex'):
449*6777b538SAndroid Build Coastguard Worker            max_dex_number = 1
450*6777b538SAndroid Build Coastguard Worker            # This is the case for incremental_install=true.
451*6777b538SAndroid Build Coastguard Worker            add_to_zip(
452*6777b538SAndroid Build Coastguard Worker                apk_dex_dir + 'classes.dex',
453*6777b538SAndroid Build Coastguard Worker                dex_file_obj.read(),
454*6777b538SAndroid Build Coastguard Worker                compress=not options.uncompress_dex)
455*6777b538SAndroid Build Coastguard Worker          else:
456*6777b538SAndroid Build Coastguard Worker            max_dex_number = 0
457*6777b538SAndroid Build Coastguard Worker            with zipfile.ZipFile(dex_file_obj) as dex_zip:
458*6777b538SAndroid Build Coastguard Worker              for dex in (d for d in dex_zip.namelist() if d.endswith('.dex')):
459*6777b538SAndroid Build Coastguard Worker                max_dex_number += 1
460*6777b538SAndroid Build Coastguard Worker                add_to_zip(
461*6777b538SAndroid Build Coastguard Worker                    apk_dex_dir + dex,
462*6777b538SAndroid Build Coastguard Worker                    dex_zip.read(dex),
463*6777b538SAndroid Build Coastguard Worker                    compress=not options.uncompress_dex)
464*6777b538SAndroid Build Coastguard Worker
465*6777b538SAndroid Build Coastguard Worker      # 4. Native libraries.
466*6777b538SAndroid Build Coastguard Worker      logging.debug('Adding lib/')
467*6777b538SAndroid Build Coastguard Worker      _AddFiles(out_apk, libs_to_add)
468*6777b538SAndroid Build Coastguard Worker
469*6777b538SAndroid Build Coastguard Worker      # Add a placeholder lib if the APK should be multi ABI but is missing libs
470*6777b538SAndroid Build Coastguard Worker      # for one of the ABIs.
471*6777b538SAndroid Build Coastguard Worker      native_lib_placeholders = options.native_lib_placeholders
472*6777b538SAndroid Build Coastguard Worker      secondary_native_lib_placeholders = (
473*6777b538SAndroid Build Coastguard Worker          options.secondary_native_lib_placeholders)
474*6777b538SAndroid Build Coastguard Worker      if options.is_multi_abi:
475*6777b538SAndroid Build Coastguard Worker        if ((secondary_native_libs or secondary_native_lib_placeholders)
476*6777b538SAndroid Build Coastguard Worker            and not native_libs and not native_lib_placeholders):
477*6777b538SAndroid Build Coastguard Worker          native_lib_placeholders += ['libplaceholder.so']
478*6777b538SAndroid Build Coastguard Worker        if ((native_libs or native_lib_placeholders)
479*6777b538SAndroid Build Coastguard Worker            and not secondary_native_libs
480*6777b538SAndroid Build Coastguard Worker            and not secondary_native_lib_placeholders):
481*6777b538SAndroid Build Coastguard Worker          secondary_native_lib_placeholders += ['libplaceholder.so']
482*6777b538SAndroid Build Coastguard Worker
483*6777b538SAndroid Build Coastguard Worker      # Add placeholder libs.
484*6777b538SAndroid Build Coastguard Worker      for name in sorted(native_lib_placeholders):
485*6777b538SAndroid Build Coastguard Worker        # Note: Empty libs files are ignored by md5check (can cause issues
486*6777b538SAndroid Build Coastguard Worker        # with stale builds when the only change is adding/removing
487*6777b538SAndroid Build Coastguard Worker        # placeholders).
488*6777b538SAndroid Build Coastguard Worker        apk_path = 'lib/%s/%s' % (options.android_abi, name)
489*6777b538SAndroid Build Coastguard Worker        alignment = _GetAbiAlignment(options.android_abi)
490*6777b538SAndroid Build Coastguard Worker        add_to_zip(apk_path, '', alignment=alignment)
491*6777b538SAndroid Build Coastguard Worker
492*6777b538SAndroid Build Coastguard Worker      for name in sorted(secondary_native_lib_placeholders):
493*6777b538SAndroid Build Coastguard Worker        # Note: Empty libs files are ignored by md5check (can cause issues
494*6777b538SAndroid Build Coastguard Worker        # with stale builds when the only change is adding/removing
495*6777b538SAndroid Build Coastguard Worker        # placeholders).
496*6777b538SAndroid Build Coastguard Worker        apk_path = 'lib/%s/%s' % (options.secondary_android_abi, name)
497*6777b538SAndroid Build Coastguard Worker        alignment = _GetAbiAlignment(options.secondary_android_abi)
498*6777b538SAndroid Build Coastguard Worker        add_to_zip(apk_path, '', alignment=alignment)
499*6777b538SAndroid Build Coastguard Worker
500*6777b538SAndroid Build Coastguard Worker      # 5. Resources
501*6777b538SAndroid Build Coastguard Worker      logging.debug('Adding res/')
502*6777b538SAndroid Build Coastguard Worker      for info in sorted(resource_infos, key=lambda i: i.filename):
503*6777b538SAndroid Build Coastguard Worker        if info.filename != 'AndroidManifest.xml':
504*6777b538SAndroid Build Coastguard Worker          copy_resource(info)
505*6777b538SAndroid Build Coastguard Worker
506*6777b538SAndroid Build Coastguard Worker      # 6. Java resources that should be accessible via
507*6777b538SAndroid Build Coastguard Worker      # Class.getResourceAsStream(), in particular parts of Emma jar.
508*6777b538SAndroid Build Coastguard Worker      # Prebuilt jars may contain class files which we shouldn't include.
509*6777b538SAndroid Build Coastguard Worker      logging.debug('Adding Java resources')
510*6777b538SAndroid Build Coastguard Worker      for java_resource in options.java_resources:
511*6777b538SAndroid Build Coastguard Worker        with zipfile.ZipFile(java_resource, 'r') as java_resource_jar:
512*6777b538SAndroid Build Coastguard Worker          for apk_path in sorted(java_resource_jar.namelist()):
513*6777b538SAndroid Build Coastguard Worker            apk_path_lower = apk_path.lower()
514*6777b538SAndroid Build Coastguard Worker
515*6777b538SAndroid Build Coastguard Worker            if apk_path_lower.startswith('meta-inf/'):
516*6777b538SAndroid Build Coastguard Worker              continue
517*6777b538SAndroid Build Coastguard Worker            if apk_path_lower.endswith('/'):
518*6777b538SAndroid Build Coastguard Worker              continue
519*6777b538SAndroid Build Coastguard Worker            if apk_path_lower.endswith('.class'):
520*6777b538SAndroid Build Coastguard Worker              continue
521*6777b538SAndroid Build Coastguard Worker
522*6777b538SAndroid Build Coastguard Worker            add_to_zip(apk_root_dir + apk_path,
523*6777b538SAndroid Build Coastguard Worker                       java_resource_jar.read(apk_path))
524*6777b538SAndroid Build Coastguard Worker
525*6777b538SAndroid Build Coastguard Worker    if options.format == 'apk' and options.key_path:
526*6777b538SAndroid Build Coastguard Worker      zipalign_path = None if fast_align else options.zipalign_path
527*6777b538SAndroid Build Coastguard Worker      finalize_apk.FinalizeApk(options.apksigner_jar,
528*6777b538SAndroid Build Coastguard Worker                               zipalign_path,
529*6777b538SAndroid Build Coastguard Worker                               f.name,
530*6777b538SAndroid Build Coastguard Worker                               f.name,
531*6777b538SAndroid Build Coastguard Worker                               options.key_path,
532*6777b538SAndroid Build Coastguard Worker                               options.key_passwd,
533*6777b538SAndroid Build Coastguard Worker                               options.key_name,
534*6777b538SAndroid Build Coastguard Worker                               int(options.min_sdk_version),
535*6777b538SAndroid Build Coastguard Worker                               warnings_as_errors=options.warnings_as_errors)
536*6777b538SAndroid Build Coastguard Worker    logging.debug('Moving file into place')
537*6777b538SAndroid Build Coastguard Worker
538*6777b538SAndroid Build Coastguard Worker    if options.depfile:
539*6777b538SAndroid Build Coastguard Worker      action_helpers.write_depfile(options.depfile,
540*6777b538SAndroid Build Coastguard Worker                                   options.output_apk,
541*6777b538SAndroid Build Coastguard Worker                                   inputs=depfile_deps)
542*6777b538SAndroid Build Coastguard Worker
543*6777b538SAndroid Build Coastguard Worker
544*6777b538SAndroid Build Coastguard Workerif __name__ == '__main__':
545*6777b538SAndroid Build Coastguard Worker  main(sys.argv[1:])
546